Commit ed439848 authored by Lars Ellenberg's avatar Lars Ellenberg Committed by Philipp Reisner

drbd: fix setsockopt for user mode linux

We use our own copy of kernel_setsockopt, and did not mess around with
get_fs/set_fs, since we thought we knew we would always be KERNEL_DS
anyways. Apparently not so for at least user mode linux, so put the
set_fs(KERNEL_DS) in there.
Signed-off-by: default avatarPhilipp Reisner <philipp.reisner@linbit.com>
Signed-off-by: default avatarLars Ellenberg <lars.ellenberg@linbit.com>
parent 71932efc
...@@ -1484,46 +1484,53 @@ static inline void drbd_flush_workqueue(struct drbd_conf *mdev) ...@@ -1484,46 +1484,53 @@ static inline void drbd_flush_workqueue(struct drbd_conf *mdev)
conn_flush_workqueue(mdev->tconn); conn_flush_workqueue(mdev->tconn);
} }
/* yes, there is kernel_setsockopt, but only since 2.6.18. we don't need to /* Yes, there is kernel_setsockopt, but only since 2.6.18.
* mess with get_fs/set_fs, we know we are KERNEL_DS always. */ * So we have our own copy of it here. */
static inline int drbd_setsockopt(struct socket *sock, int level, int optname, static inline int drbd_setsockopt(struct socket *sock, int level, int optname,
char __user *optval, int optlen) char *optval, int optlen)
{ {
mm_segment_t oldfs = get_fs();
char __user *uoptval;
int err; int err;
uoptval = (char __user __force *)optval;
set_fs(KERNEL_DS);
if (level == SOL_SOCKET) if (level == SOL_SOCKET)
err = sock_setsockopt(sock, level, optname, optval, optlen); err = sock_setsockopt(sock, level, optname, uoptval, optlen);
else else
err = sock->ops->setsockopt(sock, level, optname, optval, err = sock->ops->setsockopt(sock, level, optname, uoptval,
optlen); optlen);
set_fs(oldfs);
return err; return err;
} }
static inline void drbd_tcp_cork(struct socket *sock) static inline void drbd_tcp_cork(struct socket *sock)
{ {
int __user val = 1; int val = 1;
(void) drbd_setsockopt(sock, SOL_TCP, TCP_CORK, (void) drbd_setsockopt(sock, SOL_TCP, TCP_CORK,
(char __user *)&val, sizeof(val)); (char*)&val, sizeof(val));
} }
static inline void drbd_tcp_uncork(struct socket *sock) static inline void drbd_tcp_uncork(struct socket *sock)
{ {
int __user val = 0; int val = 0;
(void) drbd_setsockopt(sock, SOL_TCP, TCP_CORK, (void) drbd_setsockopt(sock, SOL_TCP, TCP_CORK,
(char __user *)&val, sizeof(val)); (char*)&val, sizeof(val));
} }
static inline void drbd_tcp_nodelay(struct socket *sock) static inline void drbd_tcp_nodelay(struct socket *sock)
{ {
int __user val = 1; int val = 1;
(void) drbd_setsockopt(sock, SOL_TCP, TCP_NODELAY, (void) drbd_setsockopt(sock, SOL_TCP, TCP_NODELAY,
(char __user *)&val, sizeof(val)); (char*)&val, sizeof(val));
} }
static inline void drbd_tcp_quickack(struct socket *sock) static inline void drbd_tcp_quickack(struct socket *sock)
{ {
int __user val = 2; int val = 2;
(void) drbd_setsockopt(sock, SOL_TCP, TCP_QUICKACK, (void) drbd_setsockopt(sock, SOL_TCP, TCP_QUICKACK,
(char __user *)&val, sizeof(val)); (char*)&val, sizeof(val));
} }
void drbd_bump_write_ordering(struct drbd_conf *mdev, enum write_ordering_e wo); void drbd_bump_write_ordering(struct drbd_conf *mdev, enum write_ordering_e wo);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment