pastebin - collaborative debugging tool
rovema.kpaste.net RSS


nfs_mount.exe -o port=... option patch
Posted by Anonymous on Thu 21st Sep 2023 12:45
raw | new post
modification of post by Anonymous (view diff)

  1. From 98d73158a24ddd85794925d5f6147efe30fe9333 Mon Sep 17 00:00:00 2001
  2. From: Roland Mainz <roland.mainz@nrubsig.org>
  3. Date: Thu, 21 Sep 2023 13:13:30 +0200
  4. Subject: [PATCH] nfs_mount.exe should support the -o port=portnum mount option
  5.  
  6. Implement support for the -o port=portnum option in nfs_mount.exe,
  7. specifying the numeric value of the server's NFSv4 service port.
  8. If the server's NFSv4 service is not available on the specified port,
  9. the mount request fails.
  10.  
  11. This option allows the NFS version 4.x client to contact an NFS
  12. version 4 server through a firewall, or accesing NFS version 4
  13. filesystems via ssh TCP port forwarding.
  14.  
  15. If this mount option is not specified, the NFS client uses the
  16. standard NFS TCP port number of 2049.
  17.  
  18. ** Example usage (using Cygwin 3.5.x):
  19. ---- snip ----
  20. wget 'http://svn.nrubsig.org/svn/people/gisburn/scripts/mount_sshnfs/mount_sshnfs.ksh'
  21.  
  22. mkdir /foobarmnt
  23. ksh93 ./mount_sshnfs.ksh mount -o mount_sshnfs_local_forward_port=49000 ssh+nfs://gisburn@chickenmonster/export/home/gisburn /foobarmnt
  24.  
  25. ls -l /foobarmnt
  26.  
  27. ksh93 ./mount_sshnfs.ksh umount /foobarmnt
  28. ---- snip ----
  29.  
  30. ** Limitations:
  31. - Right now only one custom port number can be used, see
  32. SAVE_PORT_HACK in sys/nfs41_driver.c. This will be fixed in the
  33. future by encoding the port number in the hostname part of the UNC
  34. path, e.g. \\server@port\nfs4\path
  35.  
  36. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  37. ---
  38. daemon/mount.c              | 12 ++++++----
  39.  daemon/namespace.c          | 10 ++++++--
  40.  daemon/nfs41.h              |  2 ++
  41.  daemon/nfs41_client.c       |  7 ++++++
  42.  daemon/nfs41_server.c       |  2 +-
  43.  daemon/upcall.h             |  1 +
  44.  libtirpc/src/pmap_getport.c |  3 ++-
  45.  mount/mount.c               |  3 ++-
  46.  sys/nfs41_driver.c          | 47 +++++++++++++++++++++++++++++++++----
  47.  9 files changed, 73 insertions(+), 14 deletions(-)
  48.  
  49. diff --git a/daemon/mount.c b/daemon/mount.c
  50. index 55345f6..3b18cba 100644
  51. --- a/daemon/mount.c
  52. +++ b/daemon/mount.c
  53. @@ -37,6 +37,8 @@ static int parse_mount(unsigned char *buffer, uint32_t length, nfs41_upcall *upc
  54.  
  55.      status = get_name(&buffer, &length, &args->hostname);
  56.      if(status) goto out;
  57. +    status = safe_read(&buffer, &length, &args->port, sizeof(DWORD));
  58. +    if (status) goto out;
  59.      status = get_name(&buffer, &length, &args->path);
  60.      if(status) goto out;
  61.      status = safe_read(&buffer, &length, &args->sec_flavor, sizeof(DWORD));
  62. @@ -46,10 +48,12 @@ static int parse_mount(unsigned char *buffer, uint32_t length, nfs41_upcall *upc
  63.      status = safe_read(&buffer, &length, &args->wsize, sizeof(DWORD));
  64.      if (status) goto out;
  65.  
  66. -    dprintf(1, "parsing NFS14_MOUNT: srv_name=%s root=%s sec_flavor=%s "
  67. -        "rsize=%d wsize=%d\n", args->hostname, args->path,
  68. +    dprintf(1, "parsing NFS41_MOUNT: srv_name=%s port=%d root=%s "
  69. +        "sec_flavor=%s rsize=%d wsize=%d\n", args->hostname, args->port, args->path,
  70.          secflavorop2name(args->sec_flavor), args->rsize, args->wsize);
  71. +    return status;
  72.  out:
  73. +    dprintf(1, "parsing NFS41_MOUNT: failed %d\n", status);
  74.      return status;
  75.  }
  76.  
  77. @@ -64,7 +68,7 @@ static int handle_mount(nfs41_upcall *upcall)
  78.      nfs41_path_fh file;
  79.  
  80.      // resolve hostname,port
  81. -    status = nfs41_server_resolve(args->hostname, 2049, &addrs);
  82. +    status = nfs41_server_resolve(args->hostname, (unsigned short)args->port, &addrs);
  83.      if (status) {
  84.          eprintf("nfs41_server_resolve() failed with %d\n", status);
  85.          goto out;
  86. @@ -76,7 +80,7 @@ static int handle_mount(nfs41_upcall *upcall)
  87.          root = upcall->root_ref;
  88.      } else {
  89.          // create root
  90. -        status = nfs41_root_create(args->hostname, args->sec_flavor,
  91. +        status = nfs41_root_create(args->hostname, args->port, args->sec_flavor,
  92.              args->wsize + WRITE_OVERHEAD, args->rsize + READ_OVERHEAD, &root);
  93.          if (status) {
  94.              eprintf("nfs41_root_create() failed %d\n", status);
  95. diff --git a/daemon/namespace.c b/daemon/namespace.c
  96. index ab16f49..e3ea077 100644
  97. --- a/daemon/namespace.c
  98. +++ b/daemon/namespace.c
  99. @@ -36,6 +36,7 @@
  100.  /* nfs41_root */
  101.  int nfs41_root_create(
  102.      IN const char *name,
  103. +    IN uint32_t port,
  104.      IN uint32_t sec_flavor,
  105.      IN uint32_t wsize,
  106.      IN uint32_t rsize,
  107. @@ -44,7 +45,7 @@ int nfs41_root_create(
  108.      int status = NO_ERROR;
  109.      nfs41_root *root;
  110.  
  111. -    dprintf(NSLVL, "--> nfs41_root_create()\n");
  112. +    dprintf(NSLVL, "--> nfs41_root_create(name=%s, port=%d)\n", name, port);
  113.  
  114.      root = calloc(1, sizeof(nfs41_root));
  115.      if (root == NULL) {
  116. @@ -60,7 +61,7 @@ int nfs41_root_create(
  117.      root->sec_flavor = sec_flavor;
  118.  
  119.      /* generate a unique client_owner */
  120. -    status = nfs41_client_owner(name, sec_flavor, &root->client_owner);
  121. +    status = nfs41_client_owner(name, port, sec_flavor, &root->client_owner);
  122.      if (status) {
  123.          eprintf("nfs41_client_owner() failed with %d\n", status);
  124.          free(root);
  125. @@ -443,6 +444,8 @@ static int referral_mount_location(
  126.      int status = ERROR_BAD_NET_NAME;
  127.      uint32_t i;
  128.  
  129. +    dprintf(NSLVL, "--> referral_mount_location()\n");
  130. +
  131.      /* create a client and session for the first available server */
  132.      for (i = 0; i < loc->server_count; i++) {
  133.          /* XXX: only deals with 'address' as a hostname with default port */
  134. @@ -453,6 +456,9 @@ static int referral_mount_location(
  135.          if (status == NO_ERROR)
  136.              break;
  137.      }
  138. +
  139. +    dprintf(NSLVL, "<-- referral_mount_location() returning %d\n", status);
  140. +
  141.      return status;
  142.  }
  143.  
  144. diff --git a/daemon/nfs41.h b/daemon/nfs41.h
  145. index aa679a6..3902393 100644
  146. --- a/daemon/nfs41.h
  147. +++ b/daemon/nfs41.h
  148. @@ -287,6 +287,7 @@ typedef struct __nfs41_root {
  149.  /* nfs41_namespace.c */
  150.  int nfs41_root_create(
  151.      IN const char *name,
  152. +    IN uint32_t port,
  153.      IN uint32_t sec_flavor,
  154.      IN uint32_t wsize,
  155.      IN uint32_t rsize,
  156. @@ -404,6 +405,7 @@ void nfs41_server_addrs(
  157.  /* nfs41_client.c */
  158.  int nfs41_client_owner(
  159.      IN const char *name,
  160. +    IN uint32_t port,
  161.      IN uint32_t sec_flavor,
  162.      OUT client_owner4 *owner);
  163.  
  164. diff --git a/daemon/nfs41_client.c b/daemon/nfs41_client.c
  165. index 0460fc5..f5bac1d 100644
  166. --- a/daemon/nfs41_client.c
  167. +++ b/daemon/nfs41_client.c
  168. @@ -353,6 +353,7 @@ out:
  169.  
  170.  int nfs41_client_owner(
  171.      IN const char *name,
  172. +    IN uint32_t port,
  173.      IN uint32_t sec_flavor,
  174.      OUT client_owner4 *owner)
  175.  {
  176. @@ -405,6 +406,12 @@ int nfs41_client_owner(
  177.          goto out_hash;
  178.      }
  179.  
  180. +    if (!CryptHashData(hash, (const BYTE*)&port, (DWORD)sizeof(port), 0)) {
  181. +        status = GetLastError();
  182. +        eprintf("CryptHashData() failed with %d\n", status);
  183. +        goto out_hash;
  184. +    }
  185. +
  186.      /* add the mac address from each applicable adapter to the hash */
  187.      status = hash_mac_addrs(hash);
  188.      if (status) {
  189. diff --git a/daemon/nfs41_server.c b/daemon/nfs41_server.c
  190. index d58c03a..ef8f0ac 100644
  191. --- a/daemon/nfs41_server.c
  192. +++ b/daemon/nfs41_server.c
  193. @@ -338,6 +338,6 @@ out:
  194.              "error %d\n", hostname, port, status);
  195.      else
  196.          dprintf(SRVLVL, "<-- nfs41_server_resolve(%s:%u) returning "
  197. -            "%s\n", hostname, port, addrs->arr[0].uaddr);
  198. +            "OK %s\n", hostname, port, addrs->arr[0].uaddr);
  199.      return status;
  200.  }
  201. diff --git a/daemon/upcall.h b/daemon/upcall.h
  202. index fd26353..9cdc0a1 100644
  203. --- a/daemon/upcall.h
  204. +++ b/daemon/upcall.h
  205. @@ -30,6 +30,7 @@
  206.  /* structures for upcall arguments */
  207.  typedef struct __mount_upcall_args {
  208.      const char *hostname;
  209. +    DWORD       port;
  210.      const char *path;
  211.      DWORD       sec_flavor;
  212.      DWORD       rsize;
  213. diff --git a/libtirpc/src/pmap_getport.c b/libtirpc/src/pmap_getport.c
  214. index 45c8b1c..4d3a277 100644
  215. --- a/libtirpc/src/pmap_getport.c
  216. +++ b/libtirpc/src/pmap_getport.c
  217. @@ -62,6 +62,7 @@ pmap_getport(address, program, version, protocol)
  218.         u_long version;
  219.         u_int protocol;
  220.  {
  221. +       (void)fprintf(stderr, "pmap_getport: returning default NFS port 2049\n");
  222.         return (u_short)2049;
  223.  }
  224.  #else
  225. @@ -106,4 +107,4 @@ pmap_getport(address, program, version, protocol)
  226.         address->sin_port = 0;
  227.         return (port);
  228.  }
  229. -#endif /* ! _WIN32 */
  230. \ No newline at end of file
  231. +#endif /* ! _WIN32 */
  232. diff --git a/mount/mount.c b/mount/mount.c
  233. index 4615fc9..6d8c8ee 100644
  234. --- a/mount/mount.c
  235. +++ b/mount/mount.c
  236. @@ -61,13 +61,14 @@ static VOID PrintUsage(LPTSTR pProcess)
  237.          TEXT("\t-o <comma-separated mount options>\n")
  238.          TEXT("Mount options:\n")
  239.          TEXT("\tro\tmount as read-only\n")
  240. +        TEXT("\tport=#\tTCP port to use (defaults to 2049)\n")
  241.          TEXT("\trsize=#\tread buffer size in bytes\n")
  242.          TEXT("\twsize=#\twrite buffer size in bytes\n")
  243.          TEXT("\tsec=krb5:krb5i:krb5p\tspecify gss security flavor\n")
  244.          TEXT("\twritethru\tturns off rdbss caching for writes\n")
  245.          TEXT("\tnocache\tturns off rdbss caching\n")
  246.          TEXT("\ttimeout=#\tspecify upcall timeout value in seconds (default 120s)\n"),
  247. -        pProcess, pProcess, pProcess);
  248. +        pProcess);
  249.  }
  250.  
  251.  DWORD __cdecl _tmain(DWORD argc, LPTSTR argv[])
  252. diff --git a/sys/nfs41_driver.c b/sys/nfs41_driver.c
  253. index da9327f..01c0cd3 100644
  254. --- a/sys/nfs41_driver.c
  255. +++ b/sys/nfs41_driver.c
  256. @@ -166,6 +166,7 @@ typedef struct _updowncall_entry {
  257.      union {
  258.          struct {
  259.              PUNICODE_STRING srv_name;
  260. +           DWORD port;
  261.              PUNICODE_STRING root;
  262.              PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs;
  263.              DWORD sec_flavor;
  264. @@ -314,6 +315,7 @@ DECLARE_CONST_UNICODE_STRING(SLASH, L"\\");
  265.  DECLARE_CONST_UNICODE_STRING(EMPTY_STRING, L"");
  266.  
  267.  #define SERVER_NAME_BUFFER_SIZE         1024
  268. +#define MOUNT_CONFIG_NFS_PORT_DEFAULT   2049
  269.  #define MOUNT_CONFIG_RW_SIZE_MIN        1024
  270.  #define MOUNT_CONFIG_RW_SIZE_DEFAULT    1048576
  271.  #define MOUNT_CONFIG_RW_SIZE_MAX        1048576
  272. @@ -328,6 +330,7 @@ typedef struct _NFS41_MOUNT_CONFIG {
  273.      BOOLEAN nocache;
  274.      WCHAR srv_buffer[SERVER_NAME_BUFFER_SIZE];
  275.      UNICODE_STRING SrvName;
  276. +    DWORD Port;
  277.      WCHAR mntpt_buffer[MAX_PATH];
  278.      UNICODE_STRING MntPt;
  279.      WCHAR sec_flavor[MAX_SEC_FLAVOR_LEN];
  280. @@ -615,13 +618,15 @@ NTSTATUS marshal_nfs41_mount(
  281.          goto out;
  282.      }
  283.      header_len = *len + length_as_utf8(entry->u.Mount.srv_name) +
  284. -        length_as_utf8(entry->u.Mount.root) + 3 * sizeof(DWORD);
  285. +        length_as_utf8(entry->u.Mount.root) + 4 * sizeof(DWORD);
  286.      if (header_len > buf_len) {
  287.          status = STATUS_INSUFFICIENT_RESOURCES;
  288.          goto out;
  289.      }
  290.      status = marshall_unicode_as_utf8(&tmp, entry->u.Mount.srv_name);
  291.      if (status) goto out;
  292. +    RtlCopyMemory(tmp, &entry->u.Mount.port, sizeof(DWORD));
  293. +    tmp += sizeof(DWORD);
  294.      status = marshall_unicode_as_utf8(&tmp, entry->u.Mount.root);
  295.      if (status) goto out;
  296.      RtlCopyMemory(tmp, &entry->u.Mount.sec_flavor, sizeof(DWORD));
  297. @@ -633,8 +638,9 @@ NTSTATUS marshal_nfs41_mount(
  298.      *len = header_len;
  299.  
  300.  #ifdef DEBUG_MARSHAL_DETAIL
  301. -    DbgP("marshal_nfs41_mount: server name=%wZ mount point=%wZ sec_flavor=%s "
  302. -         "rsize=%d wsize=%d\n", entry->u.Mount.srv_name, entry->u.Mount.root,
  303. +    DbgP("marshal_nfs41_mount: server name=%wZ port=%d mount point=%wZ "
  304. +         "sec_flavor=%s rsize=%d wsize=%d\n",
  305. +        entry->u.Mount.srv_name, entry->u.Mount.port, entry->u.Mount.root,
  306.           secflavorop2name(entry->u.Mount.sec_flavor), entry->u.Mount.rsize,
  307.           entry->u.Mount.wsize);
  308.  #endif
  309. @@ -2566,14 +2572,15 @@ NTSTATUS nfs41_mount(
  310.  
  311.  #ifdef DEBUG_MOUNT
  312.      DbgEn();
  313. -    DbgP("Server Name %wZ Mount Point %wZ SecFlavor %d\n",
  314. -        &config->SrvName, &config->MntPt, sec_flavor);
  315. +    DbgP("Server Name %wZ Port %d Mount Point %wZ SecFlavor %d\n",
  316. +        &config->SrvName, config->Port, &config->MntPt, sec_flavor);
  317.  #endif
  318.      status = nfs41_UpcallCreate(NFS41_MOUNT, NULL, *session,
  319.          INVALID_HANDLE_VALUE, *version, &config->MntPt, &entry);
  320.      if (status) goto out;
  321.  
  322.      entry->u.Mount.srv_name = &config->SrvName;
  323. +    entry->u.Mount.port = config->Port;
  324.      entry->u.Mount.root = &config->MntPt;
  325.      entry->u.Mount.rsize = config->ReadSize;
  326.      entry->u.Mount.wsize = config->WriteSize;
  327. @@ -2606,6 +2613,7 @@ void nfs41_MountConfig_InitDefaults(
  328.  {
  329.      RtlZeroMemory(Config, sizeof(NFS41_MOUNT_CONFIG));
  330.  
  331. +    Config->Port = MOUNT_CONFIG_NFS_PORT_DEFAULT;
  332.      Config->ReadSize = MOUNT_CONFIG_RW_SIZE_DEFAULT;
  333.      Config->WriteSize = MOUNT_CONFIG_RW_SIZE_DEFAULT;
  334.      Config->ReadOnly = FALSE;
  335. @@ -2724,6 +2732,11 @@ NTSTATUS nfs41_MountConfig_ParseOptions(
  336.              else
  337.                  RtlCopyUnicodeString(&Config->SrvName, &usValue);
  338.          }
  339. +        else if (wcsncmp(L"port", Name, NameLen) == 0) {
  340. +            status = nfs41_MountConfig_ParseDword(Option, &usValue,
  341. +                &Config->Port, 1,
  342. +                65535);
  343. +        }
  344.          else if (wcsncmp(L"mntpt", Name, NameLen) == 0) {
  345.              if (usValue.Length > Config->MntPt.MaximumLength)
  346.                  status = STATUS_NAME_TOO_LONG;
  347. @@ -2909,7 +2922,13 @@ NTSTATUS nfs41_CreateVNetRoot(
  348.      }
  349.      nfs41_MountConfig_InitDefaults(Config);
  350.  
  351. +#define SAVE_PORT_HACK 1
  352. +#ifdef SAVE_PORT_HACK
  353. +    static DWORD saved_port = 0;
  354. +#endif /* SAVE_PORT_HACK */
  355.      if (pCreateNetRootContext->RxContext->Create.EaLength) {
  356. +        /* Codepath for nfs_mount.exe */
  357. +
  358.          /* parse the extended attributes for mount options */
  359.          status = nfs41_MountConfig_ParseOptions(
  360.              pCreateNetRootContext->RxContext->Create.EaBuffer,
  361. @@ -2920,13 +2939,31 @@ NTSTATUS nfs41_CreateVNetRoot(
  362.          pVNetRootContext->read_only = Config->ReadOnly;
  363.          pVNetRootContext->write_thru = Config->write_thru;
  364.          pVNetRootContext->nocache = Config->nocache;        
  365. +#ifdef SAVE_PORT_HACK
  366. +       saved_port = Config->Port;
  367. +#endif /* SAVE_PORT_HACK */
  368.      } else {
  369. +        /* Codepath for \\server:port\nfs4\path */
  370. +
  371.          /* use the SRV_CALL name (without leading \) as the hostname */
  372.          Config->SrvName.Buffer = pSrvCall->pSrvCallName->Buffer + 1;
  373.          Config->SrvName.Length =
  374.              pSrvCall->pSrvCallName->Length - sizeof(WCHAR);
  375.          Config->SrvName.MaximumLength =
  376.              pSrvCall->pSrvCallName->MaximumLength - sizeof(WCHAR);
  377. +        /*
  378. +         * gisburn: FIXME: Using |saved_port| here is wrong:
  379. +         * TCP port information should be encoded in the server name
  380. +         * (e.g. \\server@port\nfs4\path, to make sure we do not get
  381. +         * conflicts in case of (ssh) port forwarding
  382. +        * (see https://learn.microsoft.com/en-gb/windows/win32/api/davclnt/nf-davclnt-davgethttpfromuncpath
  383. +        * how to specify ports in an UNC path)
  384. +         */
  385. +#ifdef SAVE_PORT_HACK
  386. +       Config->Port = saved_port;
  387. +#else
  388. +#error Getting Config->Port information from UNC server name not implemented yet
  389. +#endif /* SAVE_PORT_HACK */
  390.      }
  391.      pVNetRootContext->MountPathLen = Config->MntPt.Length;
  392.      pVNetRootContext->timeout = Config->timeout;
  393. --
  394. 2.39.0

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