auth.c 2.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697
  1. /* Copyright (C) 1995, 1996 Olaf Kirch <okir@monad.swb.de> */
  2. #include <linux/sched.h>
  3. #include <linux/user_namespace.h>
  4. #include "nfsd.h"
  5. #include "auth.h"
  6. int nfsexp_flags(struct svc_rqst *rqstp, struct svc_export *exp)
  7. {
  8. struct exp_flavor_info *f;
  9. struct exp_flavor_info *end = exp->ex_flavors + exp->ex_nflavors;
  10. for (f = exp->ex_flavors; f < end; f++) {
  11. if (f->pseudoflavor == rqstp->rq_flavor)
  12. return f->flags;
  13. }
  14. return exp->ex_flags;
  15. }
  16. int nfsd_setuser(struct svc_rqst *rqstp, struct svc_export *exp)
  17. {
  18. struct group_info *rqgi;
  19. struct group_info *gi;
  20. struct cred *new;
  21. int i;
  22. int flags = nfsexp_flags(rqstp, exp);
  23. int ret;
  24. validate_process_creds();
  25. /* discard any old override before preparing the new set */
  26. revert_creds(get_cred(current->real_cred));
  27. new = prepare_creds();
  28. if (!new)
  29. return -ENOMEM;
  30. new->fsuid = rqstp->rq_cred.cr_uid;
  31. new->fsgid = rqstp->rq_cred.cr_gid;
  32. rqgi = rqstp->rq_cred.cr_group_info;
  33. if (flags & NFSEXP_ALLSQUASH) {
  34. new->fsuid = exp->ex_anon_uid;
  35. new->fsgid = exp->ex_anon_gid;
  36. gi = groups_alloc(0);
  37. if (!gi)
  38. goto oom;
  39. } else if (flags & NFSEXP_ROOTSQUASH) {
  40. if (!new->fsuid)
  41. new->fsuid = exp->ex_anon_uid;
  42. if (!new->fsgid)
  43. new->fsgid = exp->ex_anon_gid;
  44. gi = groups_alloc(rqgi->ngroups);
  45. if (!gi)
  46. goto oom;
  47. for (i = 0; i < rqgi->ngroups; i++) {
  48. if (gid_eq(GLOBAL_ROOT_GID, GROUP_AT(rqgi, i)))
  49. GROUP_AT(gi, i) = make_kgid(&init_user_ns, exp->ex_anon_gid);
  50. else
  51. GROUP_AT(gi, i) = GROUP_AT(rqgi, i);
  52. }
  53. } else {
  54. gi = get_group_info(rqgi);
  55. }
  56. if (new->fsuid == (uid_t) -1)
  57. new->fsuid = exp->ex_anon_uid;
  58. if (new->fsgid == (gid_t) -1)
  59. new->fsgid = exp->ex_anon_gid;
  60. ret = set_groups(new, gi);
  61. put_group_info(gi);
  62. if (ret < 0)
  63. goto error;
  64. if (new->fsuid)
  65. new->cap_effective = cap_drop_nfsd_set(new->cap_effective);
  66. else
  67. new->cap_effective = cap_raise_nfsd_set(new->cap_effective,
  68. new->cap_permitted);
  69. validate_process_creds();
  70. put_cred(override_creds(new));
  71. put_cred(new);
  72. validate_process_creds();
  73. return 0;
  74. oom:
  75. ret = -ENOMEM;
  76. error:
  77. abort_creds(new);
  78. return ret;
  79. }