pastebin - collaborative debugging tool
rovema.kpaste.net RSS


msnfs41client: Patches for disk-/quota-full errors, nfs_umount.exe, cygwinaccount2nfs4account.ksh+misc, 2024-07-25
Posted by Anonymous on Thu 25th Jul 2024 20:25
raw | new post

  1. From 32277a7309e810583156e9cee8a03982b9c70571 Mon Sep 17 00:00:00 2001
  2. From: Roland Mainz <roland.mainz@nrubsig.org>
  3. Date: Thu, 25 Jul 2024 18:19:48 +0200
  4. Subject: [PATCH 1/6] cygwin,daemon,sys: Handle NFSv4 "disk full" (full, quota,
  5.  file to large) errors
  6.  
  7. Handle NFSv4 "disk full" (full, quota, file to large for filesystem etc)
  8. errors.
  9.  
  10. Also document that memory-mapped files can cause Windows event log
  11. entries ("MUP 0xc0000222" - |STATUS_LOST_WRITEBEHIND_DATA|) if
  12. a writeback from a memory-mapped fails.
  13.  
  14. Reported-by: Martin Wege <martin.l.wege@gmail.com>
  15. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  16. ---
  17. cygwin/README.bintarball.txt | 11 +++++++++++
  18.  daemon/util.c                |  1 +
  19.  sys/nfs41_driver.c           | 17 +++++++++++++++++
  20.  3 files changed, 29 insertions(+)
  21.  
  22. diff --git a/cygwin/README.bintarball.txt b/cygwin/README.bintarball.txt
  23. index 8cca4cc..8236bb6 100644
  24. --- a/cygwin/README.bintarball.txt
  25. +++ b/cygwin/README.bintarball.txt
  26. @@ -431,6 +431,17 @@ $ /sbin/nfs_mount
  27.    option, e.g.
  28.    $ /sbin/nfs_mount -o rw,writethru 'j' derfwpc5131:/export/home/rmainz #
  29.  
  30. +- Windows event log can list errors like "MUP 0xc0000222"
  31. +  (|STATUS_LOST_WRITEBEHIND_DATA|) in case the disk on the NFSv4 server
  32. +  is full and outstanding writes from a memory-mapped file fail.
  33. +  Example:
  34. +  ---- snip ----
  35. +  {Fehler beim verzoegerten Schreibvorgang} Nicht alle Daten fuer die
  36. +  Datei "\\34.159.25.153@2049\nfs4\export\nfs4export\gcc\lto-dump.exe"
  37. +  konnten gespeichert werden. Daten gingen verloren.
  38. +  Dieser Fehler wurde von dem Server zurueckgegeben, auf dem sich die
  39. +  Datei befindet. Versuchen Sie, die Datei woanders zu speichern.
  40. +  ---- snip ----
  41.  
  42.  #
  43.  # 11. Notes for troubleshooting && finding bugs/debugging:
  44. diff --git a/daemon/util.c b/daemon/util.c
  45. index 44bac60..4a7d9a9 100644
  46. --- a/daemon/util.c
  47. +++ b/daemon/util.c
  48. @@ -339,6 +339,7 @@ int nfs_to_windows_error(int status, int default_error)
  49.      case NFS4ERR_INVAL:         return ERROR_INVALID_PARAMETER;
  50.      case NFS4ERR_FBIG:          return ERROR_FILE_TOO_LARGE;
  51.      case NFS4ERR_NOSPC:         return ERROR_DISK_FULL;
  52. +    case NFS4ERR_DQUOT:         return ERROR_DISK_QUOTA_EXCEEDED;
  53.      case NFS4ERR_ROFS:          return ERROR_NETWORK_ACCESS_DENIED;
  54.      case NFS4ERR_MLINK:         return ERROR_TOO_MANY_LINKS;
  55.      case NFS4ERR_NAMETOOLONG:   return ERROR_FILENAME_EXCED_RANGE;
  56. diff --git a/sys/nfs41_driver.c b/sys/nfs41_driver.c
  57. index cc4c741..f1c2ee8 100644
  58. --- a/sys/nfs41_driver.c
  59. +++ b/sys/nfs41_driver.c
  60. @@ -3963,6 +3963,9 @@ NTSTATUS map_open_errors(
  61.      case ERROR_TOO_MANY_LINKS:          return STATUS_TOO_MANY_LINKS;
  62.      case ERROR_DIRECTORY:               return STATUS_FILE_IS_A_DIRECTORY;
  63.      case ERROR_BAD_FILE_TYPE:           return STATUS_NOT_A_DIRECTORY;
  64. +    case ERROR_DISK_FULL:               return STATUS_DISK_FULL;
  65. +    case ERROR_DISK_QUOTA_EXCEEDED:     return STATUS_DISK_QUOTA_EXCEEDED;
  66. +    case ERROR_FILE_TOO_LARGE:          return STATUS_FILE_TOO_LARGE;
  67.      case ERROR_INTERNAL_ERROR:          return STATUS_INTERNAL_ERROR;
  68.      default:
  69.          print_error("[ERROR] nfs41_Create: upcall returned ERROR_0x%x "
  70. @@ -4587,6 +4590,9 @@ NTSTATUS map_close_errors(
  71.      case ERROR_NETNAME_DELETED: return STATUS_NETWORK_NAME_DELETED;
  72.      case ERROR_NOT_EMPTY:       return STATUS_DIRECTORY_NOT_EMPTY;
  73.      case ERROR_FILE_INVALID:    return STATUS_FILE_INVALID;
  74. +    case ERROR_DISK_FULL:       return STATUS_DISK_FULL;
  75. +    case ERROR_DISK_QUOTA_EXCEEDED: return STATUS_DISK_QUOTA_EXCEEDED;
  76. +    case ERROR_FILE_TOO_LARGE:  return STATUS_FILE_TOO_LARGE;
  77.      default:
  78.          print_error("map_close_errors: "
  79.              "failed to map windows ERROR_0x%x to NTSTATUS; "
  80. @@ -5155,6 +5161,8 @@ NTSTATUS map_setea_error(
  81.      case ERROR_INVALID_EA_HANDLE:       return STATUS_NONEXISTENT_EA_ENTRY;
  82.      case ERROR_NO_MORE_FILES:           return STATUS_NO_MORE_EAS;
  83.      case ERROR_EA_FILE_CORRUPT:         return STATUS_EA_CORRUPT_ERROR;
  84. +    case ERROR_DISK_FULL:               return STATUS_DISK_FULL;
  85. +    case ERROR_DISK_QUOTA_EXCEEDED:     return STATUS_DISK_QUOTA_EXCEEDED;
  86.      case ERROR_INTERNAL_ERROR:          return STATUS_INTERNAL_ERROR;
  87.      default:
  88.          print_error("map_setea_error: "
  89. @@ -6031,6 +6039,9 @@ NTSTATUS map_setfile_error(
  90.      case ERROR_NETWORK_ACCESS_DENIED:   return STATUS_NETWORK_ACCESS_DENIED;
  91.      case ERROR_NETNAME_DELETED:         return STATUS_NETWORK_NAME_DELETED;
  92.      case ERROR_BUFFER_OVERFLOW:         return STATUS_INSUFFICIENT_RESOURCES;
  93. +    case ERROR_DISK_FULL:               return STATUS_DISK_FULL;
  94. +    case ERROR_DISK_QUOTA_EXCEEDED:     return STATUS_DISK_QUOTA_EXCEEDED;
  95. +    case ERROR_FILE_TOO_LARGE:          return STATUS_FILE_TOO_LARGE;
  96.      case ERROR_INTERNAL_ERROR:          return STATUS_INTERNAL_ERROR;
  97.      default:
  98.          print_error("map_setfile_error: "
  99. @@ -6438,6 +6449,9 @@ NTSTATUS map_readwrite_errors(
  100.      case ERROR_LOCK_VIOLATION:          return STATUS_FILE_LOCK_CONFLICT;
  101.      case ERROR_NETWORK_ACCESS_DENIED:   return STATUS_NETWORK_ACCESS_DENIED;
  102.      case ERROR_NETNAME_DELETED:         return STATUS_NETWORK_NAME_DELETED;
  103. +    case ERROR_DISK_FULL:               return STATUS_DISK_FULL;
  104. +    case ERROR_DISK_QUOTA_EXCEEDED:     return STATUS_DISK_QUOTA_EXCEEDED;
  105. +    case ERROR_FILE_TOO_LARGE:          return STATUS_FILE_TOO_LARGE;
  106.      case ERROR_INTERNAL_ERROR:          return STATUS_INTERNAL_ERROR;
  107.      default:
  108.          print_error("map_readwrite_errors: "
  109. @@ -6939,6 +6953,9 @@ NTSTATUS map_symlink_errors(
  110.      case ERROR_INSUFFICIENT_BUFFER: return STATUS_BUFFER_TOO_SMALL;
  111.      case STATUS_BUFFER_TOO_SMALL:
  112.      case ERROR_BUFFER_OVERFLOW:     return STATUS_BUFFER_OVERFLOW;
  113. +    case ERROR_DISK_FULL:           return STATUS_DISK_FULL;
  114. +    case ERROR_DISK_QUOTA_EXCEEDED: return STATUS_DISK_QUOTA_EXCEEDED;
  115. +    case ERROR_FILE_TOO_LARGE:      return STATUS_FILE_TOO_LARGE;
  116.      case ERROR_INTERNAL_ERROR:      return STATUS_INTERNAL_ERROR;
  117.      default:
  118.          print_error("map_symlink_errors: "
  119. --
  120. 2.45.1
  121.  
  122. From dd4f99c7a7089cf15736e56d485e23d3e6b6e7b1 Mon Sep 17 00:00:00 2001
  123. From: Roland Mainz <roland.mainz@nrubsig.org>
  124. Date: Thu, 25 Jul 2024 18:30:47 +0200
  125. Subject: [PATCH 2/6] cygwin,mount: Provide nfs_umount to unmount NFSv4.1
  126.  filesystems
  127.  
  128. Provide nfs_umount.exe to unmount NFSv4.1 filesystems,
  129. for feature-parity with Microsofts NFSv3 umount.exe command.
  130.  
  131. Reported-by: Josh Hurst <joshhurst@gmail.com>
  132. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  133. ---
  134. cygwin/Makefile.install      |   2 +
  135.  cygwin/README.bintarball.txt |   4 +-
  136.  mount/mount.c                | 155 ++++++++++++++++++++++++++++++-----
  137.  3 files changed, 137 insertions(+), 24 deletions(-)
  138.  
  139. diff --git a/cygwin/Makefile.install b/cygwin/Makefile.install
  140. index cf0f700..774da61 100644
  141. --- a/cygwin/Makefile.install
  142. +++ b/cygwin/Makefile.install
  143. @@ -36,6 +36,8 @@ installdest:
  144.         cp -r $(VS_BUILD_DIR)/nfsd.exe          $(DESTDIR)/$(CYGWIN_BASEPATH)/sbin/nfsd_debug.exe
  145.         cp -r $(VS_BUILD_DIR)/nfsd.pdb          $(DESTDIR)/$(CYGWIN_BASEPATH)/sbin/nfsd_debug.pdb
  146.         cp -r $(VS_BUILD_DIR)/nfs_mount.*       $(DESTDIR)/$(CYGWIN_BASEPATH)/sbin/.
  147. +       # we need a hardlink for nfs_umount.exe, softlinks do not work
  148. +       ln -f $(DESTDIR)/$(CYGWIN_BASEPATH)/sbin/nfs_mount.exe  $(DESTDIR)/$(CYGWIN_BASEPATH)/sbin/nfs_umount.exe
  149.         cp -r $(VS_BUILD_DIR)/nfsd.*            $(DESTDIR)/$(CYGWIN_BASEPATH)/sbin/.
  150.         cp -r $(VS_BUILD_DIR)/nfs_install.*     $(DESTDIR)/$(CYGWIN_BASEPATH)/sbin/.
  151.         cp -r $(VS_BUILD_DIR)/libtirpc.*        $(DESTDIR)/$(CYGWIN_BASEPATH)/sbin/.
  152. diff --git a/cygwin/README.bintarball.txt b/cygwin/README.bintarball.txt
  153. index 8236bb6..ddb43f7 100644
  154. --- a/cygwin/README.bintarball.txt
  155. +++ b/cygwin/README.bintarball.txt
  156. @@ -249,7 +249,7 @@ drwxr-xr-x 3 Unix_User+197608 Unix_Group+197121  60 Dec 13 17:58 directory_t
  157.  drwxr-xr-x 3 Unix_User+197608 Unix_Group+197121  60 Dec  7 11:01 test2
  158.  
  159.  # Unmount filesystem:
  160. -$ cd ~ && /sbin/nfs_mount -d N:
  161. +$ cd ~ && /sbin/nfs_umount N:
  162.  # OR
  163.  $ cd ~
  164.  $ net use N: /delete
  165. @@ -399,7 +399,7 @@ $ /sbin/nfs_mount
  166.    FileBothDirInformation, FileFullDirInfo, or FileIdFullDirInfo.
  167.  
  168.  - Win10/32bit-only: $ net use H: /delete # does not work,
  169. -  use $ nfs_mount -d 'H' instead #
  170. +  use $ nfs_umount 'H' instead #
  171.  
  172.  - Bug: Subversion checkout can fail with
  173.    "sqlite[S11]: database disk image is malformed" like this:
  174. diff --git a/mount/mount.c b/mount/mount.c
  175. index 9758e1c..2799980 100644
  176. --- a/mount/mount.c
  177. +++ b/mount/mount.c
  178. @@ -73,10 +73,13 @@ static BOOL ParseDriveLetter(
  179.  void PrintErrorMessage(
  180.      IN DWORD dwError);
  181.  
  182. -static VOID PrintUsage(LPWSTR pProcess)
  183. +static
  184. +void PrintMountUsage(LPWSTR pProcess)
  185.  {
  186.      (void)fprintf(stderr,
  187.          "Usage: %S [options] <drive letter|*> <hostname>:<path>\n"
  188. +        "Usage: %S -d [options] <drive letter>\n"
  189. +        "Usage: %S\n"
  190.  
  191.          "* Options:\n"
  192.          "\t-h, --help, /?\thelp\n"
  193. @@ -138,10 +141,26 @@ static VOID PrintUsage(LPWSTR pProcess)
  194.          "\tnfs_mount.exe -o sec=sys,rw S nfs://myhost1//dirwithspace/dir%%20space/test2\n"
  195.          "\tnfs_mount.exe -o sec=sys,rw S nfs://myhost1//dirwithspace/dir+space/test2\n"
  196.          "\tnfs_mount.exe -o sec=sys S nfs://myhost1//dirwithspace/dir+space/test2?rw=1\n",
  197. -        pProcess, (int)NFS41_DRIVER_DEFAULT_CREATE_MODE);
  198. +        pProcess, pProcess, pProcess,
  199. +        (int)NFS41_DRIVER_DEFAULT_CREATE_MODE);
  200.  }
  201.  
  202. -int __cdecl wmain(int argc, wchar_t *argv[])
  203. +
  204. +static
  205. +void PrintUmountUsage(LPWSTR pProcess)
  206. +{
  207. +    (void)fprintf(stderr,
  208. +        "Usage: %S [options] <drive letter>\n"
  209. +
  210. +        "* Options:\n"
  211. +        "\t-h, --help, /?\thelp\n"
  212. +        "\t-f, --force\tforce unmount if the drive is in use\n",
  213. +        pProcess);
  214. +}
  215. +
  216. +
  217. +static
  218. +int mount_main(int argc, wchar_t *argv[])
  219.  {
  220.      int     i;
  221.      DWORD   result = NO_ERROR;
  222. @@ -157,16 +176,6 @@ int __cdecl wmain(int argc, wchar_t *argv[])
  223.      wchar_t *mntopts[MAX_MNTOPTS] = { 0 };
  224.      int     num_mntopts = 0;
  225.  
  226. -    int crtsetdbgflags = 0;
  227. -    crtsetdbgflags |= _CRTDBG_ALLOC_MEM_DF;  /* use debug heap */
  228. -    crtsetdbgflags |= _CRTDBG_LEAK_CHECK_DF; /* report leaks on exit */
  229. -    crtsetdbgflags |= _CRTDBG_DELAY_FREE_MEM_DF;
  230. -    (void)_CrtSetDbgFlag(crtsetdbgflags);
  231. -    (void)_CrtSetReportMode(_CRT_WARN, _CRTDBG_MODE_FILE);
  232. -    (void)_CrtSetReportMode(_CRT_ERROR, _CRTDBG_MODE_FILE);
  233. -    (void)_CrtSetReportMode(_CRT_ASSERT, _CRTDBG_MODE_FILE);
  234. -
  235. -
  236.      if (argc == 1) {
  237.          /* list open nfs shares */
  238.          result = EnumMounts(NULL);
  239. @@ -191,7 +200,7 @@ int __cdecl wmain(int argc, wchar_t *argv[])
  240.              /* help */
  241.              if ((!wcscmp(argv[i], L"-h")) ||
  242.                  (!wcscmp(argv[i], L"--help"))) {
  243. -                PrintUsage(argv[0]);
  244. +                PrintMountUsage(argv[0]);
  245.                  goto out;
  246.              }
  247.              /* unmount */
  248. @@ -218,7 +227,7 @@ int __cdecl wmain(int argc, wchar_t *argv[])
  249.                      result = ERROR_BAD_ARGUMENTS;
  250.                      (void)fwprintf(stderr,
  251.                          L"Mount options missing after '-o'.\n\n");
  252. -                    PrintUsage(argv[0]);
  253. +                    PrintMountUsage(argv[0]);
  254.                      goto out_free;
  255.                  }
  256.  
  257. @@ -335,15 +344,15 @@ opt_o_argv_i_again:
  258.                      result = ERROR_BAD_ARGUMENTS;
  259.                      (void)fwprintf(stderr, L"Filesystem type missing "
  260.                          L"after '-t'/'-F'.\n\n");
  261. -                    PrintUsage(argv[0]);
  262. +                    PrintMountUsage(argv[0]);
  263.                      goto out_free;
  264.                  }
  265.  
  266.                  if (!wcscmp(argv[i], L"nfs")) {
  267.                      result = ERROR_BAD_ARGUMENTS;
  268.                      (void)fwprintf(stderr, L"Filesystem type '%s' "
  269. -                        L"not supported.\n\n", argv[i]);
  270. -                    PrintUsage(argv[0]);
  271. +                        L"not supported.\n\n./build.vc19/x64/Debug/nfs_mount.exe", argv[i]);
  272. +                    PrintMountUsage(argv[0]);
  273.                      goto out_free;
  274.                  }
  275.              }
  276. @@ -354,7 +363,7 @@ opt_o_argv_i_again:
  277.          }
  278.          /* Windows-style "nfs_mount /?" help */
  279.          else if (!wcscmp(argv[i], L"/?")) {
  280. -            PrintUsage(argv[0]);
  281. +            PrintMountUsage(argv[0]);
  282.              goto out;
  283.         }
  284.          /* drive letter */
  285. @@ -377,7 +386,7 @@ opt_o_argv_i_again:
  286.      {
  287.          result = ERROR_BAD_ARGUMENTS;
  288.          (void)fwprintf(stderr, L"Missing argument for drive letter.\n\n");
  289. -        PrintUsage(argv[0]);
  290. +        PrintMountUsage(argv[0]);
  291.          goto out_free;
  292.      }
  293.      if (FALSE == ParseDriveLetter(pLocalName, szLocalName))
  294. @@ -386,7 +395,7 @@ opt_o_argv_i_again:
  295.          (void)fwprintf(stderr, L"Invalid drive letter '%s'. "
  296.              L"Expected 'C' or 'C:'.\n\n",
  297.              pLocalName);
  298. -        PrintUsage(argv[0]);
  299. +        PrintMountUsage(argv[0]);
  300.          goto out_free;
  301.      }
  302.  
  303. @@ -407,7 +416,7 @@ opt_o_argv_i_again:
  304.          {
  305.              result = ERROR_BAD_NET_NAME;
  306.              (void)fwprintf(stderr, L"Missing argument for remote path.\n\n");
  307. -            PrintUsage(argv[0]);
  308. +            PrintMountUsage(argv[0]);
  309.              goto out_free;
  310.          }
  311.  
  312. @@ -446,6 +455,108 @@ out:
  313.      return result;
  314.  }
  315.  
  316. +
  317. +static
  318. +int umount_main(int argc, wchar_t *argv[])
  319. +{
  320. +    int     i;
  321. +    DWORD   result = NO_ERROR;
  322. +    LPWSTR  pLocalName = NULL;
  323. +    wchar_t szLocalName[] = L"C:\0";
  324. +    BOOL    bForceUnmount = FALSE;
  325. +
  326. +    /* parse command line */
  327. +    for (i = 1; i < argc; i++) {
  328. +        if (argv[i][0] == L'-') {
  329. +            /* help */
  330. +            if ((!wcscmp(argv[i], L"-h")) ||
  331. +                (!wcscmp(argv[i], L"--help"))) {
  332. +                PrintUmountUsage(argv[0]);
  333. +                goto out;
  334. +            }
  335. +            /* force unmount */
  336. +            else if ((!wcscmp(argv[i], L"-f")) ||
  337. +                (!wcscmp(argv[i], L"--force"))) {
  338. +                bForceUnmount = TRUE;
  339. +            }
  340. +            else {
  341. +                (void)fwprintf(stderr, L"Unrecognized option "
  342. +                    L"'%s', disregarding.\n",
  343. +                    argv[i]);
  344. +                result = ERROR_BAD_ARGUMENTS;
  345. +            }
  346. +        }
  347. +        /* Windows-style "nfs_umount /?" help */
  348. +        else if (!wcscmp(argv[i], L"/?")) {
  349. +            PrintUmountUsage(argv[0]);
  350. +            goto out;
  351. +       }
  352. +        /* drive letter */
  353. +       else if (pLocalName == NULL) {
  354. +            pLocalName = argv[i];
  355. +        }
  356. +        else {
  357. +            (void)fwprintf(stderr, L"Unrecognized argument "
  358. +                L"'%s', disregarding.\n",
  359. +                argv[i]);
  360. +        }
  361. +    }
  362. +
  363. +    if (pLocalName == NULL) {
  364. +        result = ERROR_BAD_ARGUMENTS;
  365. +        (void)fwprintf(stderr, L"Drive letter expected.\n");
  366. +        PrintUmountUsage(argv[0]);
  367. +        goto out;
  368. +    }
  369. +
  370. +    if (!ParseDriveLetter(pLocalName, szLocalName)) {
  371. +        result = ERROR_BAD_ARGUMENTS;
  372. +        (void)fwprintf(stderr, L"Invalid drive letter '%s'. "
  373. +            L"Expected 'C' or 'C:'.\n\n",
  374. +            pLocalName);
  375. +        PrintUmountUsage(argv[0]);
  376. +        goto out;
  377. +    }
  378. +
  379. +    result = DoUnmount(szLocalName, bForceUnmount);
  380. +    if (result)
  381. +        PrintErrorMessage(result);
  382. +out:
  383. +    return result;
  384. +}
  385. +
  386. +
  387. +int __cdecl wmain(int argc, wchar_t *argv[])
  388. +{
  389. +    DWORD result = NO_ERROR;
  390. +
  391. +    int crtsetdbgflags = 0;
  392. +    crtsetdbgflags |= _CRTDBG_ALLOC_MEM_DF;  /* use debug heap */
  393. +    crtsetdbgflags |= _CRTDBG_LEAK_CHECK_DF; /* report leaks on exit */
  394. +    crtsetdbgflags |= _CRTDBG_DELAY_FREE_MEM_DF;
  395. +    (void)_CrtSetDbgFlag(crtsetdbgflags);
  396. +    (void)_CrtSetReportMode(_CRT_WARN, _CRTDBG_MODE_FILE);
  397. +    (void)_CrtSetReportMode(_CRT_ERROR, _CRTDBG_MODE_FILE);
  398. +    (void)_CrtSetReportMode(_CRT_ASSERT, _CRTDBG_MODE_FILE);
  399. +
  400. +    if (wcsstr(argv[0], L"nfs_mount")) {
  401. +        result = mount_main(argc, argv);
  402. +        goto out;
  403. +    }
  404. +    else if (wcsstr(argv[0], L"nfs_umount")) {
  405. +        result = umount_main(argc, argv);
  406. +        goto out;
  407. +    }
  408. +    else {
  409. +        (void)fwprintf(stderr, L"%s: Unknown mode\n", argv[0]);
  410. +        result = 1;
  411. +        goto out;
  412. +    }
  413. +out:
  414. +    return result;
  415. +}
  416. +
  417. +
  418.  static void ConvertUnixSlashes(
  419.      IN OUT LPWSTR pRemoteName)
  420.  {
  421. --
  422. 2.45.1
  423.  
  424. From c12fe659a0dca9e72fe4178b24260df91d04e03a Mon Sep 17 00:00:00 2001
  425. From: Roland Mainz <roland.mainz@nrubsig.org>
  426. Date: Thu, 25 Jul 2024 18:40:41 +0200
  427. Subject: [PATCH 3/6] sys: |map_symlink_errors()| should handle
  428.  |ERROR_TOO_MANY_LINKS|
  429.  
  430. |map_symlink_errors()| should map |ERROR_TOO_MANY_LINKS| to
  431. |STATUS_TOO_MANY_LINKS|.
  432.  
  433. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  434. ---
  435. sys/nfs41_driver.c | 1 +
  436.  1 file changed, 1 insertion(+)
  437.  
  438. diff --git a/sys/nfs41_driver.c b/sys/nfs41_driver.c
  439. index f1c2ee8..0cf7316 100644
  440. --- a/sys/nfs41_driver.c
  441. +++ b/sys/nfs41_driver.c
  442. @@ -6956,6 +6956,7 @@ NTSTATUS map_symlink_errors(
  443.      case ERROR_DISK_FULL:           return STATUS_DISK_FULL;
  444.      case ERROR_DISK_QUOTA_EXCEEDED: return STATUS_DISK_QUOTA_EXCEEDED;
  445.      case ERROR_FILE_TOO_LARGE:      return STATUS_FILE_TOO_LARGE;
  446. +    case ERROR_TOO_MANY_LINKS:      return STATUS_TOO_MANY_LINKS;
  447.      case ERROR_INTERNAL_ERROR:      return STATUS_INTERNAL_ERROR;
  448.      default:
  449.          print_error("map_symlink_errors: "
  450. --
  451. 2.45.1
  452.  
  453. From b32b1baa038885039cac38e58762044ea6d54750 Mon Sep 17 00:00:00 2001
  454. From: Roland Mainz <roland.mainz@nrubsig.org>
  455. Date: Thu, 25 Jul 2024 19:14:57 +0200
  456. Subject: [PATCH 4/6] mount: nfs_mount.exe exit codes should be useable
  457.  
  458. Fix nfs_mount.exe exit codes.
  459. POSIX process return values can only in the range from
  460. |0|...|SCHAR_MAX|, so returning Windows |ERROR_*| values
  461. does not work. Instead we now return { 0, 1 }, and maybe
  462. more values (up to |SCHAR_MAX-1|) in the future
  463.  
  464. Reported-by: Josh Hurst <joshhurst@gmail.com>
  465. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  466. ---
  467. mount/mount.c | 11 ++++++++++-
  468.  1 file changed, 10 insertions(+), 1 deletion(-)
  469.  
  470. diff --git a/mount/mount.c b/mount/mount.c
  471. index 2799980..6d7435c 100644
  472. --- a/mount/mount.c
  473. +++ b/mount/mount.c
  474. @@ -201,6 +201,7 @@ int mount_main(int argc, wchar_t *argv[])
  475.              if ((!wcscmp(argv[i], L"-h")) ||
  476.                  (!wcscmp(argv[i], L"--help"))) {
  477.                  PrintMountUsage(argv[0]);
  478. +                result = 1;
  479.                  goto out;
  480.              }
  481.              /* unmount */
  482. @@ -364,6 +365,7 @@ opt_o_argv_i_again:
  483.          /* Windows-style "nfs_mount /?" help */
  484.          else if (!wcscmp(argv[i], L"/?")) {
  485.              PrintMountUsage(argv[0]);
  486. +            result = 1;
  487.              goto out;
  488.         }
  489.          /* drive letter */
  490. @@ -472,6 +474,7 @@ int umount_main(int argc, wchar_t *argv[])
  491.              if ((!wcscmp(argv[i], L"-h")) ||
  492.                  (!wcscmp(argv[i], L"--help"))) {
  493.                  PrintUmountUsage(argv[0]);
  494. +                result = 1;
  495.                  goto out;
  496.              }
  497.              /* force unmount */
  498. @@ -489,6 +492,7 @@ int umount_main(int argc, wchar_t *argv[])
  499.          /* Windows-style "nfs_umount /?" help */
  500.          else if (!wcscmp(argv[i], L"/?")) {
  501.              PrintUmountUsage(argv[0]);
  502. +            result = 1;
  503.              goto out;
  504.         }
  505.          /* drive letter */
  506. @@ -552,8 +556,13 @@ int __cdecl wmain(int argc, wchar_t *argv[])
  507.          result = 1;
  508.          goto out;
  509.      }
  510. +
  511.  out:
  512. -    return result;
  513. +    /*
  514. +     * POSIX return value of a command can only in the range from
  515. +     * |0|...|SCHAR_MAX|, so map the |ERROR_*| to |1|,|0|.
  516. +     */
  517. +    return (result != NO_ERROR)?1:0;
  518.  }
  519.  
  520.  
  521. --
  522. 2.45.1
  523.  
  524. From c67e95bf7336a7c25e7d02988d05196228c420fa Mon Sep 17 00:00:00 2001
  525. From: Roland Mainz <roland.mainz@nrubsig.org>
  526. Date: Thu, 25 Jul 2024 20:33:40 +0200
  527. Subject: [PATCH 5/6] cygwin: Add utility to convert Win32 account+group info
  528.  into NFSv4 server account data
  529.  
  530. Add new /sbin/cygwinaccount2nfs4account utility to convert Win32
  531. account+group info into UNIX/Linux NFSv4 server account data
  532.  
  533. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  534. ---
  535. cygwin/Makefile.install                       |   4 +
  536.  cygwin/README.bintarball.txt                  |   6 +
  537.  .../cygwinaccount2nfs4account.ksh             | 235 ++++++++++++++++++
  538.  3 files changed, 245 insertions(+)
  539.  create mode 100644 cygwin/utils/cygwinaccount2nfs4account/cygwinaccount2nfs4account.ksh
  540.  
  541. diff --git a/cygwin/Makefile.install b/cygwin/Makefile.install
  542. index 774da61..68f2b63 100644
  543. --- a/cygwin/Makefile.install
  544. +++ b/cygwin/Makefile.install
  545. @@ -55,6 +55,10 @@ installdest:
  546.         git diff -w     >"$(DESTDIR)/$(CYGWIN_BASEPATH)/usr/src/msnfs41client/msnfs41client_diff_w.diff"
  547.         git diff        >"$(DESTDIR)/$(CYGWIN_BASEPATH)/usr/src/msnfs41client/msnfs41client_diff.diff"
  548.         @ printf "# Package utilties\n"
  549. +       cp $(CYGWIN_MAKEFILE_DIR)/utils/cygwinaccount2nfs4account/cygwinaccount2nfs4account.ksh $(DESTDIR)/$(CYGWIN_BASEPATH)/sbin/cygwinaccount2nfs4account
  550. +       chmod a+x $(DESTDIR)/$(CYGWIN_BASEPATH)/sbin/cygwinaccount2nfs4account
  551. +       PATH+=":$(DESTDIR)/$(CYGWIN_BASEPATH)/sbin/" \
  552. +               /usr/bin/ksh93 $(CYGWIN_MAKEFILE_DIR)/utils/cygwinaccount2nfs4account/cygwinaccount2nfs4account.ksh --nroff 2>"$(DESTDIR)/$(CYGWIN_BASEPATH)/usr/share/man/man1/cygwinaccount2nfs4account.1" || true
  553.         cp $(CYGWIN_MAKEFILE_DIR)/utils/mount_sshnfs/mount_sshnfs.ksh $(DESTDIR)/$(CYGWIN_BASEPATH)/sbin/mount_sshnfs
  554.         chmod a+x $(DESTDIR)/$(CYGWIN_BASEPATH)/sbin/mount_sshnfs
  555.         PATH+=":$(DESTDIR)/$(CYGWIN_BASEPATH)/sbin/" \
  556. diff --git a/cygwin/README.bintarball.txt b/cygwin/README.bintarball.txt
  557. index ddb43f7..dcbbf65 100644
  558. --- a/cygwin/README.bintarball.txt
  559. +++ b/cygwin/README.bintarball.txt
  560. @@ -350,6 +350,12 @@ $ /sbin/nfs_mount
  561.    Solaris/Illumos using export option "resvport" (see nfs(5)), as the
  562.    NFSv4 client source TCP port will be >= 1024.
  563.  
  564. +- Install: Adding Windows accounts+groups to the NFSv4 server:
  565. +  ms-nfs41-client comes with /sbin/cygwinaccount2nfs4account to
  566. +  convert the Win32/Cygwin account information of the (current)
  567. +  user+groups to a small script for the NFSv4 server to set-up
  568. +  these accounts on the server side.
  569. +
  570.  #
  571.  # 10. Known issues:
  572.  #
  573. diff --git a/cygwin/utils/cygwinaccount2nfs4account/cygwinaccount2nfs4account.ksh b/cygwin/utils/cygwinaccount2nfs4account/cygwinaccount2nfs4account.ksh
  574. new file mode 100644
  575. index 0000000..5979eb9
  576. --- /dev/null
  577. +++ b/cygwin/utils/cygwinaccount2nfs4account/cygwinaccount2nfs4account.ksh
  578. @@ -0,0 +1,235 @@
  579. +#!/bin/ksh93
  580. +
  581. +#
  582. +# MIT License
  583. +#
  584. +# Copyright (c) 2024 Roland Mainz <roland.mainz@nrubsig.org>
  585. +#
  586. +# Permission is hereby granted, free of charge, to any person obtaining a copy
  587. +# of this software and associated documentation files (the "Software"), to deal
  588. +# in the Software without restriction, including without limitation the rights
  589. +# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  590. +# copies of the Software, and to permit persons to whom the Software is
  591. +# furnished to do so, subject to the following conditions:
  592. +#
  593. +# The above copyright notice and this permission notice shall be included in all
  594. +# copies or substantial portions of the Software.
  595. +#
  596. +# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  597. +# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  598. +# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  599. +# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  600. +# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  601. +# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  602. +# SOFTWARE.
  603. +#
  604. +
  605. +#
  606. +# cygwinaccount2nfs4account.ksh93 - transfer Cygwin user/group account
  607. +# info to Linux/UNIX NFSv4 server account data
  608. +#
  609. +
  610. +#
  611. +# Written by Roland Mainz <roland.mainz@nrubsig.org>
  612. +#
  613. +
  614. +function getent_passwd2compound
  615. +{
  616. +       set -o nounset
  617. +
  618. +       typeset username="$2"
  619. +       typeset leftover
  620. +       nameref data="$1" # output compound variable
  621. +
  622. +       compound out
  623. +
  624. +       # capture getent output
  625. +       out.stderr="${ { out.stdout="${ getent passwd "$username" ; (( out.res=$? )) ; }" ; } 2>&1 ; }"
  626. +
  627. +       if [[ "${out.stderr}" != '' ]] || (( out.res != 0 )) ; then
  628. +               print -u2 $"%s: getent failed, msg=%q, res=%d\n" \
  629. +                       "$0" "${out.stderr}" out.res
  630. +               return 1
  631. +       fi
  632. +
  633. +       # ~(E) is POSIX extended regular expression matching (instead
  634. +       # of shell pattern), "x" means "multiline", "l" means "left
  635. +       # anchor", "r" means "right anchor"
  636. +       leftover="${out.stdout/~(Elrx)
  637. +               (.+?)                   # login name
  638. +               :(.+?)                  # encrypted passwd
  639. +               :(.+?)                  # uid
  640. +               :(.+?)                  # gid
  641. +               :(.+?)                  # comment
  642. +               :(.+?)                  # homedir
  643. +               (?::(.+?))?             # shell (optional)
  644. +               /X}"
  645. +
  646. +       # All parsed data should be captured via eregex in .sh.match - if
  647. +       # there is anything left (except the 'X') then the input string did
  648. +       # not properly match the eregex
  649. +       [[ "$leftover" == 'X' ]] ||
  650. +               { print -u2 -f $"%s: Parser error, leftover=%q\n" \
  651. +                       "$0" "$leftover" ; return 1 ; }
  652. +
  653. +       data.getent_username="$username"
  654. +       data.login_name="${.sh.match[1]}"
  655. +       data.encrypted_passwd="${.sh.match[2]}"
  656. +       data.uid="${.sh.match[3]}"
  657. +       data.gid="${.sh.match[4]}"
  658. +       data.comment="${.sh.match[5]}"
  659. +       data.homedir="${.sh.match[6]}"
  660. +       data.shell="${.sh.match[7]}"
  661. +
  662. +       return 0
  663. +}
  664. +
  665. +function getent_group2compound
  666. +{
  667. +       set -o nounset
  668. +
  669. +       typeset groupname="$2"
  670. +       typeset leftover
  671. +       nameref data="$1" # output compound variable
  672. +
  673. +       compound out
  674. +
  675. +       # capture getent output
  676. +       out.stderr="${ { out.stdout="${ getent group "$groupname" ; (( out.res=$? )) ; }" ; } 2>&1 ; }"
  677. +
  678. +       if [[ "${out.stderr}" != '' ]] || (( out.res != 0 )) ; then
  679. +               print -u2 $"%s: getent failed, msg=%q, res=%d\n" \
  680. +                       "$0" "${out.stderr}" out.res
  681. +               return 1
  682. +       fi
  683. +
  684. +       # ~(E) is POSIX extended regular expression matching (instead
  685. +       # of shell pattern), "x" means "multiline", "l" means "left
  686. +       # anchor", "r" means "right anchor"
  687. +       leftover="${out.stdout/~(Elrx)
  688. +               (.+?):                  # group
  689. +               (.+?):                  # encrypted passwd
  690. +               (.+?):                  # gid
  691. +               (?:(.+?))?              # userlist
  692. +               /X}"
  693. +
  694. +       # All parsed data should be captured via eregex in .sh.match - if
  695. +       # there is anything left (except the 'X') then the input string did
  696. +       # not properly match the eregex
  697. +       [[ "$leftover" == 'X' ]] ||
  698. +               { print -u2 -f $"%s: Parser error, leftover=%q\n" \
  699. +                       "$0" "$leftover" ; return 1 ; }
  700. +
  701. +       data.getent_groupname="$groupname"
  702. +       data.group_name="${.sh.match[1]}"
  703. +       data.encrypted_passwd="${.sh.match[2]}"
  704. +       data.gid="${.sh.match[3]}"
  705. +       [[ -v .sh.match[4] ]] && data.userlist="${.sh.match[4]}"
  706. +
  707. +       return 0
  708. +}
  709. +
  710. +
  711. +function accountdata2linuxscript
  712. +{
  713. +       set -o nounset
  714. +
  715. +       nameref accountdata=$1
  716. +       typeset gidlist=''
  717. +       typeset sidname
  718. +
  719. +       #
  720. +       # first start with the groups, as useradd wants a group to be there
  721. +       #
  722. +       integer i
  723. +
  724. +       for ((i=0 ; i < ${#accountdata.group_list[@]} ; i++ )) ; do
  725. +               nameref currgrp=accountdata.group_list[$i]
  726. +
  727. +               #
  728. +               # Cygwin was Win32 SID values in "encrypted_passwd",
  729. +               # we use this to reject groups which have special
  730. +               # functions in Win32
  731. +               sidname="${currgrp.encrypted_passwd}"
  732. +               # check if "sidname" is realy a Win32 SID
  733. +               if [[ "$sidname" == ~(El)S-1-[[:digit:]]+-[[:digit:]]+ ]] ; then
  734. +                       if [[ "$sidname" != ~(Elr)S-1-5-21-.+ ]] ; then
  735. +                               continue
  736. +                       fi
  737. +               fi
  738. +
  739. +               printf 'groupadd -g %s %q\n' "${currgrp.gid}" "${currgrp.group_name}"
  740. +
  741. +               [[ "$gidlist" != '' ]] && gidlist+=','
  742. +               gidlist+="${currgrp.gid}"
  743. +       done
  744. +
  745. +       #
  746. +       # user data itself
  747. +       #
  748. +       nameref curruser=accountdata.user
  749. +
  750. +       printf 'mkdir -p %q\n' "${curruser.homedir}"
  751. +       printf 'useradd -u %s -g %s -G %q -s %q %q\n' \
  752. +               "${curruser.uid}" \
  753. +               "${curruser.gid}" \
  754. +               "${gidlist}" \
  755. +               "${curruser.shell}" \
  756. +               "${curruser.login_name}"
  757. +       printf 'chown %q %q\n' \
  758. +               "${curruser.uid}:${curruser.gid}" \
  759. +               "${curruser.homedir}"
  760. +
  761. +       return 0
  762. +}
  763. +
  764. +
  765. +function convert_curruser2linuxscript
  766. +{
  767. +       compound account_data
  768. +       compound account_data.user
  769. +       compound -a account_data.group_list
  770. +       integer i=0
  771. +       typeset currgroup
  772. +
  773. +       getent_passwd2compound account_data.user "$(id -u)"
  774. +
  775. +       for currgroup in $(id -G) ; do
  776. +               getent_group2compound account_data.group_list[$((i++))] "$currgroup"
  777. +       done
  778. +
  779. +       print -v account_data
  780. +
  781. +       accountdata2linuxscript account_data
  782. +
  783. +       # fixme: we need to figure out the real NFSv4 idmapping domain of the client
  784. +       printf 'printf "Domain = GLOBAL.LOC\\n" >>"/etc/idmapd.conf"\n'
  785. +
  786. +       printf 'printf "options nfsd nfs4_disable_idmapping=N\\noptions nfs nfs4_disable_idmapping=N\\n" >>"/etc/modprobe.d/nfs.conf"\n'
  787. +       printf 'printf "NEED_IDMAPD=yes\\n" >>"/etc/default/nfs-common"\n'
  788. +
  789. +       return 0
  790. +}
  791. +
  792. +
  793. +#
  794. +# ToDo:
  795. +# - Command-line options
  796. +# - Convert current user+groups to Linux bash script [done]
  797. +# - Convert current user+groups to /etc/passwd+/etc/group lines
  798. +# - Convert given user+groups to Linux bash script
  799. +# - Convert given user+groups to /etc/passwd+/etc/group lines
  800. +#
  801. +function main
  802. +{
  803. +       convert_curruser2linuxscript "$@"
  804. +       return $?
  805. +}
  806. +
  807. +builtin id
  808. +
  809. +main "$@"
  810. +return $?
  811. +
  812. +
  813. +# EOF.
  814. --
  815. 2.45.1
  816.  
  817. From b48de242a7769fcaba7b698ac4aac8480e952694 Mon Sep 17 00:00:00 2001
  818. From: Roland Mainz <roland.mainz@nrubsig.org>
  819. Date: Thu, 25 Jul 2024 21:09:27 +0200
  820. Subject: [PATCH 6/6] cygwin: cygwinaccount2nfs4account: Put "Domain" in
  821.  "[General]" section
  822.  
  823. cygwinaccount2nfs4account: Put "Domain" in "[General]" section
  824.  
  825. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  826. ---
  827. .../cygwinaccount2nfs4account.ksh                          | 7 ++++++-
  828.  1 file changed, 6 insertions(+), 1 deletion(-)
  829.  
  830. diff --git a/cygwin/utils/cygwinaccount2nfs4account/cygwinaccount2nfs4account.ksh b/cygwin/utils/cygwinaccount2nfs4account/cygwinaccount2nfs4account.ksh
  831. index 5979eb9..f765276 100644
  832. --- a/cygwin/utils/cygwinaccount2nfs4account/cygwinaccount2nfs4account.ksh
  833. +++ b/cygwin/utils/cygwinaccount2nfs4account/cygwinaccount2nfs4account.ksh
  834. @@ -203,7 +203,12 @@ function convert_curruser2linuxscript
  835.         accountdata2linuxscript account_data
  836.  
  837.         # fixme: we need to figure out the real NFSv4 idmapping domain of the client
  838. -       printf 'printf "Domain = GLOBAL.LOC\\n" >>"/etc/idmapd.conf"\n'
  839. +
  840. +       printf '# turn idmapper on, even for AUTH_SYS\n'
  841. +       printf '{\n'
  842. +       printf '\tprintf "[General]\\n"\n'
  843. +       printf '\tprintf "Domain = %s\\n"\n' "GLOBAL.LOC"
  844. +       printf '} >>"/etc/idmapd.conf"\n'
  845.  
  846.         printf 'printf "options nfsd nfs4_disable_idmapping=N\\noptions nfs nfs4_disable_idmapping=N\\n" >>"/etc/modprobe.d/nfs.conf"\n'
  847.         printf 'printf "NEED_IDMAPD=yes\\n" >>"/etc/default/nfs-common"\n'
  848. --
  849. 2.45.1

Submit a correction or amendment below (click here to make a fresh posting)
After submitting an amendment, you'll be able to view the differences between the old and new posts easily.

Syntax highlighting:

To highlight particular lines, prefix each line with {%HIGHLIGHT}




All content is user-submitted.
The administrators of this site (kpaste.net) are not responsible for their content.
Abuse reports should be emailed to us at