|
@@ -60,46 +60,57 @@ void nf_unregister_sockopt(struct nf_sockopt_ops *reg)
|
|
}
|
|
}
|
|
EXPORT_SYMBOL(nf_unregister_sockopt);
|
|
EXPORT_SYMBOL(nf_unregister_sockopt);
|
|
|
|
|
|
-/* Call get/setsockopt() */
|
|
|
|
-static int nf_sockopt(struct sock *sk, int pf, int val,
|
|
|
|
- char __user *opt, int *len, int get)
|
|
|
|
|
|
+static struct nf_sockopt_ops *nf_sockopt_find(struct sock *sk, int pf,
|
|
|
|
+ int val, int get)
|
|
{
|
|
{
|
|
struct nf_sockopt_ops *ops;
|
|
struct nf_sockopt_ops *ops;
|
|
- int ret;
|
|
|
|
|
|
|
|
if (sk->sk_net != &init_net)
|
|
if (sk->sk_net != &init_net)
|
|
- return -ENOPROTOOPT;
|
|
|
|
|
|
+ return ERR_PTR(-ENOPROTOOPT);
|
|
|
|
|
|
if (mutex_lock_interruptible(&nf_sockopt_mutex) != 0)
|
|
if (mutex_lock_interruptible(&nf_sockopt_mutex) != 0)
|
|
- return -EINTR;
|
|
|
|
|
|
+ return ERR_PTR(-EINTR);
|
|
|
|
|
|
list_for_each_entry(ops, &nf_sockopts, list) {
|
|
list_for_each_entry(ops, &nf_sockopts, list) {
|
|
if (ops->pf == pf) {
|
|
if (ops->pf == pf) {
|
|
if (!try_module_get(ops->owner))
|
|
if (!try_module_get(ops->owner))
|
|
goto out_nosup;
|
|
goto out_nosup;
|
|
|
|
+
|
|
if (get) {
|
|
if (get) {
|
|
- if (val >= ops->get_optmin
|
|
|
|
- && val < ops->get_optmax) {
|
|
|
|
- mutex_unlock(&nf_sockopt_mutex);
|
|
|
|
- ret = ops->get(sk, val, opt, len);
|
|
|
|
|
|
+ if (val >= ops->get_optmin &&
|
|
|
|
+ val < ops->get_optmax)
|
|
goto out;
|
|
goto out;
|
|
- }
|
|
|
|
} else {
|
|
} else {
|
|
- if (val >= ops->set_optmin
|
|
|
|
- && val < ops->set_optmax) {
|
|
|
|
- mutex_unlock(&nf_sockopt_mutex);
|
|
|
|
- ret = ops->set(sk, val, opt, *len);
|
|
|
|
|
|
+ if (val >= ops->set_optmin &&
|
|
|
|
+ val < ops->set_optmax)
|
|
goto out;
|
|
goto out;
|
|
- }
|
|
|
|
}
|
|
}
|
|
module_put(ops->owner);
|
|
module_put(ops->owner);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
- out_nosup:
|
|
|
|
|
|
+out_nosup:
|
|
|
|
+ ops = ERR_PTR(-ENOPROTOOPT);
|
|
|
|
+out:
|
|
mutex_unlock(&nf_sockopt_mutex);
|
|
mutex_unlock(&nf_sockopt_mutex);
|
|
- return -ENOPROTOOPT;
|
|
|
|
|
|
+ return ops;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* Call get/setsockopt() */
|
|
|
|
+static int nf_sockopt(struct sock *sk, int pf, int val,
|
|
|
|
+ char __user *opt, int *len, int get)
|
|
|
|
+{
|
|
|
|
+ struct nf_sockopt_ops *ops;
|
|
|
|
+ int ret;
|
|
|
|
+
|
|
|
|
+ ops = nf_sockopt_find(sk, pf, val, get);
|
|
|
|
+ if (IS_ERR(ops))
|
|
|
|
+ return PTR_ERR(ops);
|
|
|
|
+
|
|
|
|
+ if (get)
|
|
|
|
+ ret = ops->get(sk, val, opt, len);
|
|
|
|
+ else
|
|
|
|
+ ret = ops->set(sk, val, opt, *len);
|
|
|
|
|
|
- out:
|
|
|
|
module_put(ops->owner);
|
|
module_put(ops->owner);
|
|
return ret;
|
|
return ret;
|
|
}
|
|
}
|
|
@@ -124,51 +135,22 @@ static int compat_nf_sockopt(struct sock *sk, int pf, int val,
|
|
struct nf_sockopt_ops *ops;
|
|
struct nf_sockopt_ops *ops;
|
|
int ret;
|
|
int ret;
|
|
|
|
|
|
- if (sk->sk_net != &init_net)
|
|
|
|
- return -ENOPROTOOPT;
|
|
|
|
-
|
|
|
|
-
|
|
|
|
- if (mutex_lock_interruptible(&nf_sockopt_mutex) != 0)
|
|
|
|
- return -EINTR;
|
|
|
|
-
|
|
|
|
- list_for_each_entry(ops, &nf_sockopts, list) {
|
|
|
|
- if (ops->pf == pf) {
|
|
|
|
- if (!try_module_get(ops->owner))
|
|
|
|
- goto out_nosup;
|
|
|
|
-
|
|
|
|
- if (get) {
|
|
|
|
- if (val >= ops->get_optmin
|
|
|
|
- && val < ops->get_optmax) {
|
|
|
|
- mutex_unlock(&nf_sockopt_mutex);
|
|
|
|
- if (ops->compat_get)
|
|
|
|
- ret = ops->compat_get(sk,
|
|
|
|
- val, opt, len);
|
|
|
|
- else
|
|
|
|
- ret = ops->get(sk,
|
|
|
|
- val, opt, len);
|
|
|
|
- goto out;
|
|
|
|
- }
|
|
|
|
- } else {
|
|
|
|
- if (val >= ops->set_optmin
|
|
|
|
- && val < ops->set_optmax) {
|
|
|
|
- mutex_unlock(&nf_sockopt_mutex);
|
|
|
|
- if (ops->compat_set)
|
|
|
|
- ret = ops->compat_set(sk,
|
|
|
|
- val, opt, *len);
|
|
|
|
- else
|
|
|
|
- ret = ops->set(sk,
|
|
|
|
- val, opt, *len);
|
|
|
|
- goto out;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- module_put(ops->owner);
|
|
|
|
- }
|
|
|
|
|
|
+ ops = nf_sockopt_find(sk, pf, val, get);
|
|
|
|
+ if (IS_ERR(ops))
|
|
|
|
+ return PTR_ERR(ops);
|
|
|
|
+
|
|
|
|
+ if (get) {
|
|
|
|
+ if (ops->compat_get)
|
|
|
|
+ ret = ops->compat_get(sk, val, opt, len);
|
|
|
|
+ else
|
|
|
|
+ ret = ops->get(sk, val, ops, len);
|
|
|
|
+ } else {
|
|
|
|
+ if (ops->compat_set)
|
|
|
|
+ ret = ops->compat_set(sk, val, ops, *len);
|
|
|
|
+ else
|
|
|
|
+ ret = ops->set(sk, val, ops, *len);
|
|
}
|
|
}
|
|
- out_nosup:
|
|
|
|
- mutex_unlock(&nf_sockopt_mutex);
|
|
|
|
- return -ENOPROTOOPT;
|
|
|
|
|
|
|
|
- out:
|
|
|
|
module_put(ops->owner);
|
|
module_put(ops->owner);
|
|
return ret;
|
|
return ret;
|
|
}
|
|
}
|