pastebin - collaborative debugging tool
rovema.kpaste.net RSS


msnfs41client: Patches for URL parameters+misc, 2024-07-09
Posted by Anonymous on Tue 9th Jul 2024 17:21
raw | new post

  1. From 4b1f90d7375a27b707d5cc3abfdfd624db5d49b6 Mon Sep 17 00:00:00 2001
  2. From: Roland Mainz <roland.mainz@nrubsig.org>
  3. Date: Tue, 9 Jul 2024 13:45:00 +0200
  4. Subject: [PATCH 1/3] cygwin,mount: Add support for (nfs://) URL parameters
  5.  
  6. Add support for (nfs://) URL parameters.
  7.  
  8. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  9. ---
  10. cygwin/utils/mount_sshnfs/mount_sshnfs.ksh |  69 ++++-
  11.  cygwin/utils/nfsurlconv/nfsurlconv.ksh     |  83 +++++-
  12.  cygwin/utils/sshnfs/sshnfs.ksh             |  59 ++++-
  13.  mount/mount.c                              |  73 ++++-
  14.  mount/urlparser1.c                         | 294 ++++++++++++++-------
  15.  mount/urlparser1.h                         |  15 +-
  16.  6 files changed, 479 insertions(+), 114 deletions(-)
  17.  
  18. diff --git a/cygwin/utils/mount_sshnfs/mount_sshnfs.ksh b/cygwin/utils/mount_sshnfs/mount_sshnfs.ksh
  19. index c1dc4b5..f323382 100644
  20. --- a/cygwin/utils/mount_sshnfs/mount_sshnfs.ksh
  21. +++ b/cygwin/utils/mount_sshnfs/mount_sshnfs.ksh
  22. @@ -240,6 +240,32 @@ function netstat_find_next_free_local_tcp_port
  23.  }
  24.  
  25.  
  26. +function urldecodestr
  27. +{
  28. +       nameref out=$1
  29. +       typeset s="$2"
  30. +
  31. +       #
  32. +       # build format string for printf(1) ...
  33. +       #
  34. +
  35. +       # quote backslashes
  36. +       s="${s//$'\\'/$'\\\\'}"
  37. +       # urldecode '+' to ' '
  38. +       s="${s//+/ }"
  39. +       # urldecode %<hexdigit><hexdigit>
  40. +       s="${s//~(E)(?:%([[:xdigit:]][[:xdigit:]]))/\\x}"
  41. +       # quote any remaining "%" to make it safe for printf(1)
  42. +       s="${s//%/%%}"
  43. +
  44. +       #
  45. +       # ... and then let printf(1) do the formatting
  46. +       #
  47. +       out="${ printf "$s" ; }"
  48. +       return 0
  49. +}
  50. +
  51. +
  52.  #
  53.  # parse_rfc1738_url - parse RFC 1838 URLs
  54.  #
  55. @@ -253,6 +279,7 @@ function parse_rfc1738_url
  56.         typeset url="$2"
  57.         typeset leftover
  58.         nameref data="$1" # output compound variable
  59. +       typeset url_param_str
  60.  
  61.         # ~(E) is POSIX extended regular expression matching (instead
  62.         # of shell pattern), "x" means "multiline", "l" means "left
  63. @@ -271,7 +298,9 @@ function parse_rfc1738_url
  64.                                 (?::([[:digit:]]+))? # port (optional)
  65.                         )
  66.                 )
  67. -               (?:\/(.*?))?/X}"                # path (optional)
  68. +               (?:\/(.*?))?                    # path (optional)
  69. +               (?:\?(.*?))?                    # URL parameters (optional)
  70. +               /X}"
  71.  
  72.         # All parsed data should be captured via eregex in .sh.match - if
  73.         # there is anything left (except the 'X') then the input string did
  74. @@ -291,8 +320,34 @@ function parse_rfc1738_url
  75.         [[ "${.sh.match[7]-}" != '' ]] && integer data.port="${.sh.match[7]}"
  76.         [[ "${.sh.match[8]-}" != '' ]] && data.uripath="${.sh.match[8]}"
  77.  
  78. +       if [[ "${.sh.match[9]-}" != '' ]] ; then
  79. +               compound -a data.parameters
  80. +
  81. +               url_param_str="${.sh.match[9]-}"
  82. +
  83. +               while [[ "$url_param_str" != '' ]] ; do
  84. +                       leftover="${url_param_str/~(Elrx)(?:(.+?)(?:=(.+?))?)(?:&(.*))?/X}"
  85. +
  86. +                       # save matches because urldecodestr uses .sh.match, too
  87. +                       typeset dp_name="${.sh.match[1]-}"
  88. +                       typeset dp_value="${.sh.match[2]-}"
  89. +                       typeset dp_next="${.sh.match[3]-}"
  90. +
  91. +                       urldecodestr dp_name "${dp_name}"
  92. +                       urldecodestr dp_value "${dp_value}"
  93. +
  94. +                       data.parameters+=(
  95. +                               name="${dp_name}"
  96. +                               value="${dp_value}"
  97. +                               )
  98. +
  99. +                       # next parameter
  100. +                       url_param_str="${dp_next}"
  101. +               done
  102. +       fi
  103. +
  104.         if [[ -v data.uripath ]] ; then
  105. -               data.path="${ printf "${data.uripath//~(E)(?:%([[:xdigit:]][[:xdigit:]]))/\\x}" ; }"
  106. +               urldecodestr data.path "${data.uripath}"
  107.         fi
  108.  
  109.         return 0
  110. @@ -367,7 +422,7 @@ function cmd_mount
  111.  
  112.         # fixme: Need better text layout for $ mount_sshnfs mount --man #
  113.         typeset -r mount_sshnfs_cmdmount_usage=$'+
  114. -       [-?\n@(#)\$Id: mount_sshnfs mount (Roland Mainz) 2024-01-31 \$\n]
  115. +       [-?\n@(#)\$Id: mount_sshnfs mount (Roland Mainz) 2024-07-08 \$\n]
  116.         [-author?Roland Mainz <roland.mainz@nrubsig.org>]
  117.         [+NAME?mount_sshnfs mount - mount NFSv4 filesystem through ssh
  118.                 tunnel]
  119. @@ -472,8 +527,8 @@ function cmd_mount
  120.                 if [[ "$s" == ~(Elr)mount_sshnfs.+=.+ ]] ; then
  121.                         case "$s" in
  122.                                 ~(Eli)mount_sshnfs_jumphost=)
  123. -                                        [[ ! -v c.ssh_jumphost_args ]] && typeset -a c.ssh_jumphost_args
  124. -                                        c.ssh_jumphost_args+=( "-J" "${c.mount_nfs_options[i]/~(Eli)mount_sshnfs_jumphost=}" )
  125. +                                       [[ ! -v c.ssh_jumphost_args ]] && typeset -a c.ssh_jumphost_args
  126. +                                       c.ssh_jumphost_args+=( "-J" "${c.mount_nfs_options[i]/~(Eli)mount_sshnfs_jumphost=}" )
  127.                                         ;;
  128.                                 ~(Eli)mount_sshnfs_local_forward_port=)
  129.                                         # command(1) prevents that the shell interpreter
  130. @@ -733,7 +788,7 @@ function cmd_umount
  131.         typeset mydebug=false # fixme: should be "bool" for ksh93v
  132.         # fixme: Need better text layout for $ mount_sshnfs mount --man #
  133.         typeset -r mount_sshnfs_cmdumount_usage=$'+
  134. -       [-?\n@(#)\$Id: mount_sshnfs umount (Roland Mainz) 2024-01-31 \$\n]
  135. +       [-?\n@(#)\$Id: mount_sshnfs umount (Roland Mainz) 2024-07-08 \$\n]
  136.         [-author?Roland Mainz <roland.mainz@nrubsig.org>]
  137.         [+NAME?mount_sshnfs umount - unmount NFSv4 filesystem mounted
  138.                 via mount_sshnfs mount]
  139. @@ -837,7 +892,7 @@ function main
  140.  
  141.         # fixme: Need better text layout for $ mount_sshnfs --man #
  142.         typeset -r mount_sshnfs_usage=$'+
  143. -       [-?\n@(#)\$Id: mount_sshnfs (Roland Mainz) 2024-01-31 \$\n]
  144. +       [-?\n@(#)\$Id: mount_sshnfs (Roland Mainz) 2024-07-08 \$\n]
  145.         [-author?Roland Mainz <roland.mainz@nrubsig.org>]
  146.         [+NAME?mount_sshnfs - mount/umount NFSv4 filesystem via ssh
  147.                 tunnel]
  148. diff --git a/cygwin/utils/nfsurlconv/nfsurlconv.ksh b/cygwin/utils/nfsurlconv/nfsurlconv.ksh
  149. index 97db401..e11c019 100644
  150. --- a/cygwin/utils/nfsurlconv/nfsurlconv.ksh
  151. +++ b/cygwin/utils/nfsurlconv/nfsurlconv.ksh
  152. @@ -38,6 +38,32 @@ function usage
  153.         return 2
  154.  }
  155.  
  156. +function urldecodestr
  157. +{
  158. +       nameref out=$1
  159. +       typeset s="$2"
  160. +
  161. +       #
  162. +       # build format string for printf(1) ...
  163. +       #
  164. +
  165. +       # quote backslashes
  166. +       s="${s//$'\\'/$'\\\\'}"
  167. +       # urldecode '+' to ' '
  168. +       s="${s//+/ }"
  169. +       # urldecode %<hexdigit><hexdigit>
  170. +       s="${s//~(E)(?:%([[:xdigit:]][[:xdigit:]]))/\\x}"
  171. +       # quote any remaining "%" to make it safe for printf(1)
  172. +       s="${s//%/%%}"
  173. +
  174. +       #
  175. +       # ... and then let printf(1) do the formatting
  176. +       #
  177. +       out="${ printf "$s" ; }"
  178. +       return 0
  179. +}
  180. +
  181. +
  182.  #
  183.  # parse_rfc1738_url - parse RFC 1838 URLs
  184.  #
  185. @@ -51,6 +77,7 @@ function parse_rfc1738_url
  186.         typeset url="$2"
  187.         typeset leftover
  188.         nameref data="$1" # output compound variable
  189. +       typeset url_param_str
  190.  
  191.         # ~(E) is POSIX extended regular expression matching (instead
  192.         # of shell pattern), "x" means "multiline", "l" means "left
  193. @@ -69,7 +96,9 @@ function parse_rfc1738_url
  194.                                 (?::([[:digit:]]+))? # port (optional)
  195.                         )
  196.                 )
  197. -               (?:\/(.*?))?/X}"                # path (optional)
  198. +               (?:\/(.*?))?                    # path (optional)
  199. +               (?:\?(.*?))?                    # URL parameters (optional)
  200. +               /X}"
  201.  
  202.         # All parsed data should be captured via eregex in .sh.match - if
  203.         # there is anything left (except the 'X') then the input string did
  204. @@ -89,8 +118,34 @@ function parse_rfc1738_url
  205.         [[ "${.sh.match[7]-}" != '' ]] && integer data.port="${.sh.match[7]}"
  206.         [[ "${.sh.match[8]-}" != '' ]] && data.uripath="${.sh.match[8]}"
  207.  
  208. +       if [[ "${.sh.match[9]-}" != '' ]] ; then
  209. +               compound -a data.parameters
  210. +
  211. +               url_param_str="${.sh.match[9]-}"
  212. +
  213. +               while [[ "$url_param_str" != '' ]] ; do
  214. +                       leftover="${url_param_str/~(Elrx)(?:(.+?)(?:=(.+?))?)(?:&(.*))?/X}"
  215. +
  216. +                       # save matches because urldecodestr uses .sh.match, too
  217. +                       typeset dp_name="${.sh.match[1]-}"
  218. +                       typeset dp_value="${.sh.match[2]-}"
  219. +                       typeset dp_next="${.sh.match[3]-}"
  220. +
  221. +                       urldecodestr dp_name "${dp_name}"
  222. +                       urldecodestr dp_value "${dp_value}"
  223. +
  224. +                       data.parameters+=(
  225. +                               name="${dp_name}"
  226. +                               value="${dp_value}"
  227. +                               )
  228. +
  229. +                       # next parameter
  230. +                       url_param_str="${dp_next}"
  231. +               done
  232. +       fi
  233. +
  234.         if [[ -v data.uripath ]] ; then
  235. -               data.path="${ printf "${data.uripath//~(E)(?:%([[:xdigit:]][[:xdigit:]]))/\\x}" ; }"
  236. +               urldecodestr data.path "${data.uripath}"
  237.         fi
  238.  
  239.         return 0
  240. @@ -207,7 +262,7 @@ function main
  241.  
  242.         # fixme: Need better text layout for $ nfsurlconv --man #
  243.         typeset -r nfsurlconv_usage=$'+
  244. -       [-?\n@(#)\$Id: nfsurlconv (Roland Mainz) 2024-01-31 \$\n]
  245. +       [-?\n@(#)\$Id: nfsurlconv (Roland Mainz) 2024-07-08 \$\n]
  246.         [-author?Roland Mainz <roland.mainz@nrubsig.org>]
  247.         [+NAME?nfsurlconv - convert hostname,port,path from/to a nfs://-URL]
  248.         [+DESCRIPTION?\bnfsurlconv\b convert { hostname, port, path } from/to a nfs://-URL.]
  249. @@ -238,6 +293,14 @@ path=/a/b/c
  250.  hostport=bbb
  251.  path=/a/b/c
  252.  ]
  253. +}
  254. +               [+?Example 4:][+?Convert URL nfs://bbb:12049//a/b/c?param1=pvalue1&param2=pvalue2 to ( hostport=, path=, urlparameter= )]{
  255. +[+\n$ nfsurlconv.ksh url2hostportpath "nfs:://bbb::12049//a/b/c??param1=pvalue1&param2=pvalue2"
  256. +hostport=bbb::12049
  257. +path=/a/b/c
  258. +urlparameter=( name=param1 value=pvalue1 )
  259. +urlparameter=( name=param2 value=pvalue2 )
  260. +]
  261.  }
  262.         }
  263.         [+SEE ALSO?\bksh93\b(1),\bssh\b(1),\bmount.nfs\b(8),\bnfs\b(5)]
  264. @@ -299,6 +362,13 @@ path=/a/b/c
  265.                         printf 'hostname=%s\n' "${urldata.host}"
  266.                         printf 'port=%s\n' "${urldata.port-2049}"
  267.                         printf 'path=%s\n' "${urldata.path-}"
  268. +                       if [[ -v urldata.parameters ]] ; then
  269. +                               for (( i=0 ; i < ${#urldata.parameters[@]} ; i++ )) ; do
  270. +                                       printf 'urlparameter=( name=%q value=%q )\n' \
  271. +                                               "${urldata.parameters[i].name}" \
  272. +                                               "${urldata.parameters[i].value}"
  273. +                               done
  274. +                       fi
  275.                         return 0
  276.                         ;;
  277.                 'url2hostportpath')
  278. @@ -307,6 +377,13 @@ path=/a/b/c
  279.                         parse_sshnfs_url urldata "${@:2:1}" || return 1
  280.                         printf 'hostport=%s\n' "${urldata.hostport}"
  281.                         printf 'path=%s\n' "${urldata.path-}"
  282. +                       if [[ -v urldata.parameters ]] ; then
  283. +                               for (( i=0 ; i < ${#urldata.parameters[@]} ; i++ )) ; do
  284. +                                       printf 'urlparameter=( name=%q value=%q )\n' \
  285. +                                               "${urldata.parameters[i].name}" \
  286. +                                               "${urldata.parameters[i].value}"
  287. +                               done
  288. +                       fi
  289.                         return 0
  290.                         ;;
  291.                 'url2compound')
  292. diff --git a/cygwin/utils/sshnfs/sshnfs.ksh b/cygwin/utils/sshnfs/sshnfs.ksh
  293. index bfccabb..29f5dc0 100644
  294. --- a/cygwin/utils/sshnfs/sshnfs.ksh
  295. +++ b/cygwin/utils/sshnfs/sshnfs.ksh
  296. @@ -222,6 +222,32 @@ function netstat_find_next_free_local_tcp_port
  297.  }
  298.  
  299.  
  300. +function urldecodestr
  301. +{
  302. +       nameref out=$1
  303. +       typeset s="$2"
  304. +
  305. +       #
  306. +       # build format string for printf(1) ...
  307. +       #
  308. +
  309. +       # quote backslashes
  310. +       s="${s//$'\\'/$'\\\\'}"
  311. +       # urldecode '+' to ' '
  312. +       s="${s//+/ }"
  313. +       # urldecode %<hexdigit><hexdigit>
  314. +       s="${s//~(E)(?:%([[:xdigit:]][[:xdigit:]]))/\\x}"
  315. +       # quote any remaining "%" to make it safe for printf(1)
  316. +       s="${s//%/%%}"
  317. +
  318. +       #
  319. +       # ... and then let printf(1) do the formatting
  320. +       #
  321. +       out="${ printf "$s" ; }"
  322. +       return 0
  323. +}
  324. +
  325. +
  326.  #
  327.  # parse_rfc1738_url - parse RFC 1838 URLs
  328.  #
  329. @@ -235,6 +261,7 @@ function parse_rfc1738_url
  330.         typeset url="$2"
  331.         typeset leftover
  332.         nameref data="$1" # output compound variable
  333. +       typeset url_param_str
  334.  
  335.         # ~(E) is POSIX extended regular expression matching (instead
  336.         # of shell pattern), "x" means "multiline", "l" means "left
  337. @@ -253,7 +280,9 @@ function parse_rfc1738_url
  338.                                 (?::([[:digit:]]+))? # port (optional)
  339.                         )
  340.                 )
  341. -               (?:\/(.*?))?/X}"                # path (optional)
  342. +               (?:\/(.*?))?                    # path (optional)
  343. +               (?:\?(.*?))?                    # URL parameters (optional)
  344. +               /X}"
  345.  
  346.         # All parsed data should be captured via eregex in .sh.match - if
  347.         # there is anything left (except the 'X') then the input string did
  348. @@ -273,8 +302,34 @@ function parse_rfc1738_url
  349.         [[ "${.sh.match[7]-}" != '' ]] && integer data.port="${.sh.match[7]}"
  350.         [[ "${.sh.match[8]-}" != '' ]] && data.uripath="${.sh.match[8]}"
  351.  
  352. +       if [[ "${.sh.match[9]-}" != '' ]] ; then
  353. +               compound -a data.parameters
  354. +
  355. +               url_param_str="${.sh.match[9]-}"
  356. +
  357. +               while [[ "$url_param_str" != '' ]] ; do
  358. +                       leftover="${url_param_str/~(Elrx)(?:(.+?)(?:=(.+?))?)(?:&(.*))?/X}"
  359. +
  360. +                       # save matches because urldecodestr uses .sh.match, too
  361. +                       typeset dp_name="${.sh.match[1]-}"
  362. +                       typeset dp_value="${.sh.match[2]-}"
  363. +                       typeset dp_next="${.sh.match[3]-}"
  364. +
  365. +                       urldecodestr dp_name "${dp_name}"
  366. +                       urldecodestr dp_value "${dp_value}"
  367. +
  368. +                       data.parameters+=(
  369. +                               name="${dp_name}"
  370. +                               value="${dp_value}"
  371. +                               )
  372. +
  373. +                       # next parameter
  374. +                       url_param_str="${dp_next}"
  375. +               done
  376. +       fi
  377. +
  378.         if [[ -v data.uripath ]] ; then
  379. -               data.path="${ printf "${data.uripath//~(E)(?:%([[:xdigit:]][[:xdigit:]]))/\\x}" ; }"
  380. +               urldecodestr data.path "${data.uripath}"
  381.         fi
  382.  
  383.         return 0
  384. diff --git a/mount/mount.c b/mount/mount.c
  385. index 67b011a..06a0c35 100644
  386. --- a/mount/mount.c
  387. +++ b/mount/mount.c
  388. @@ -71,6 +71,7 @@ static VOID PrintUsage(LPTSTR pProcess)
  389.  {
  390.      (void)_tprintf(
  391.          TEXT("Usage: %s [options] <drive letter|*> <hostname>:<path>\n")
  392. +
  393.          TEXT("* Options:\n")
  394.          TEXT("\t-h\thelp\n")
  395.          TEXT("\t/?\thelp\n")
  396. @@ -82,6 +83,7 @@ static VOID PrintUsage(LPTSTR pProcess)
  397.             " (Linux compat)\n")
  398.          TEXT("\t-p\tmake the mount persist over reboots\n")
  399.          TEXT("\t-o <comma-separated mount options>\n")
  400. +
  401.          TEXT("* Mount options:\n")
  402.          TEXT("\tro\tmount as read-only\n")
  403.          TEXT("\trw\tmount as read-write (default)\n")
  404. @@ -100,23 +102,33 @@ static VOID PrintUsage(LPTSTR pProcess)
  405.              "\t\tif this value is prefixed with 'nfsv3attrmode+'\n"
  406.              "\t\tthe mode value from a \"NfsV3Attributes\" EA will be used\n"
  407.              "\t\t(defaults \"nfsv3attrmode+0o%o\").\n")
  408. +
  409. +        TEXT("* URL parameters:\n")
  410. +        TEXT("\tro=1\tmount as read-only\n")
  411. +        TEXT("\trw=1\tmount as read-write (default)\n")
  412. +
  413.          TEXT("* Hostname:\n")
  414.          TEXT("\tDNS name, or hostname in domain\n")
  415.          TEXT("\tentry in C:\\Windows\\System32\\drivers\\etc\\hosts\n")
  416.          TEXT("\tIPv4 address\n")
  417.          TEXT("\tIPv6 address within '[', ']' "
  418.              "(will be converted to *.ipv6-literal.net)\n")
  419. +
  420.          TEXT("* Examples:\n")
  421.          TEXT("\tnfs_mount.exe -p -o rw 'H' derfwpc5131_ipv4:/export/home2/rmainz\n")
  422.          TEXT("\tnfs_mount.exe -o rw '*' bigramhost:/tmp\n")
  423. +        TEXT("\tnfs_mount.exe -o ro '*' archive1:/tmp\n")
  424. +        TEXT("\tnfs_mount.exe '*' archive1:/tmp?ro=1\n")
  425.          TEXT("\tnfs_mount.exe -o rw,sec=sys,port=30000 T grendel:/net_tmpfs2\n")
  426.          TEXT("\tnfs_mount.exe -o sec=sys,rw S nfs://myhost1//net_tmpfs2/test2\n")
  427. +        TEXT("\tnfs_mount.exe -o sec=sys S nfs://myhost1//net_tmpfs2/test2?rw=1\n")
  428.          TEXT("\tnfs_mount.exe -o sec=sys,rw S nfs://myhost1:1234//net_tmpfs2/test2\n")
  429.          TEXT("\tnfs_mount.exe -o sec=sys,rw,port=1234 S nfs://myhost1//net_tmpfs2/test2\n")
  430.          TEXT("\tnfs_mount.exe -o sec=sys,rw '*' [fe80::21b:1bff:fec3:7713]://net_tmpfs2/test2\n")
  431.          TEXT("\tnfs_mount.exe -o sec=sys,rw '*' nfs://[fe80::21b:1bff:fec3:7713]//net_tmpfs2/test2\n")
  432.          TEXT("\tnfs_mount.exe -o sec=sys,rw S nfs://myhost1//dirwithspace/dir%%20space/test2\n")
  433. -        TEXT("\tnfs_mount.exe -o sec=sys,rw S nfs://myhost1//dirwithspace/dir+space/test2\n"),
  434. +        TEXT("\tnfs_mount.exe -o sec=sys,rw S nfs://myhost1//dirwithspace/dir+space/test2\n")
  435. +        TEXT("\tnfs_mount.exe -o sec=sys S nfs://myhost1//dirwithspace/dir+space/test2?rw=1\n"),
  436.          pProcess, (int)NFS41_DRIVER_DEFAULT_CREATE_MODE);
  437.  }
  438.  
  439. @@ -404,6 +416,65 @@ static DWORD ParseRemoteName(
  440.              goto out;
  441.          }
  442.  
  443. +        if (uctx->num_parameters > 0) {
  444. +            int pi;
  445. +            const char *pname;
  446. +            const char *pvalue;
  447. +
  448. +            /*
  449. +             * FIXME: Values added here based on URL parameters
  450. +             * should be added at the front of the list of options,
  451. +             * so users can override the nfs://-URL given default.
  452. +             * Right now this does not work, e.g.
  453. +             * $ nfs_mount.exe -o rw nfs://foo//bar?ro=1 # will
  454. +             * result in a read-only mount, while the expectation
  455. +             * is that -o overrides URL settings.
  456. +             */
  457. +            for (pi = 0; pi < uctx->num_parameters ; pi++) {
  458. +                pname = uctx->parameters[pi].name;
  459. +                pvalue = uctx->parameters[pi].value;
  460. +
  461. +                if (!strcmp(pname, "rw")) {
  462. +                    if ((pvalue == NULL) || (!strcmp(pvalue, "1"))) {
  463. +                        (void)InsertOption(TEXT("rw"), TEXT("1"), pOptions);
  464. +                    }
  465. +                    else if (!strcmp(pvalue, "0")) {
  466. +                        (void)InsertOption(TEXT("ro"), TEXT("1"), pOptions);
  467. +                    }
  468. +                    else {
  469. +                        result = ERROR_BAD_ARGUMENTS;
  470. +                        (void)_ftprintf(stderr,
  471. +                            TEXT("Unsupported nfs://-URL parameter ")
  472. +                            TEXT("'%S' value '%S'.\n"),
  473. +                            pname, pvalue);
  474. +                        goto out;
  475. +                    }
  476. +                }
  477. +                else if (!strcmp(pname, "ro")) {
  478. +                    if ((pvalue == NULL) || (!strcmp(pvalue, "1"))) {
  479. +                        (void)InsertOption(TEXT("ro"), TEXT("1"), pOptions);
  480. +                    }
  481. +                    else if (!strcmp(pvalue, "0")) {
  482. +                        (void)InsertOption(TEXT("rw"), TEXT("1"), pOptions);
  483. +                    }
  484. +                    else {
  485. +                        result = ERROR_BAD_ARGUMENTS;
  486. +                        (void)_ftprintf(stderr,
  487. +                            TEXT("Unsupported nfs://-URL parameter ")
  488. +                            TEXT("'%S' value '%S'.\n"),
  489. +                            pname, pvalue);
  490. +                        goto out;
  491. +                    }
  492. +                }
  493. +                else {
  494. +                    result = ERROR_BAD_ARGUMENTS;
  495. +                    (void)_ftprintf(stderr,
  496. +                        TEXT("Unsupported nfs://-URL parameter '%S'.\n"), pname);
  497. +                    goto out;
  498. +                }
  499. +            }
  500. +        }
  501. +
  502.          if (uctx->hostport.port != -1)
  503.              port = uctx->hostport.port;
  504.  
  505. diff --git a/mount/urlparser1.c b/mount/urlparser1.c
  506. index a90d1e2..8fcae52 100644
  507. --- a/mount/urlparser1.c
  508. +++ b/mount/urlparser1.c
  509. @@ -28,10 +28,23 @@
  510.  #include <stdlib.h>
  511.  #include <stdbool.h>
  512.  #include <string.h>
  513. +#include <ctype.h>
  514.  #include <stdio.h>
  515.  
  516. +// #define TEST_URLPARSER 1
  517. +
  518.  #include "urlparser1.h"
  519.  
  520. +typedef struct _url_parser_context_private {
  521. +       url_parser_context c;
  522. +
  523. +       /* Private data */
  524. +       char *parameter_string_buff;
  525. +} url_parser_context_private;
  526. +
  527. +#define MAX_URL_PARAMETERS 256
  528. +
  529. +#ifdef _MSC_VER
  530.  /*
  531.   * Disable "warning C4996: 'wcscpy': This function or variable may be
  532.   * unsafe." because in this case the buffers are properly sized,
  533. @@ -43,88 +56,81 @@
  534.   * because it is safe to use in our code.
  535.   */
  536.  #pragma warning (disable : 4706)
  537. +#endif /* _MSC_VER */
  538.  
  539.  /*
  540.   * Original extended regular expression:
  541.   *
  542.   * "^"
  543. - * "(.+?)"                      // scheme
  544. - * "://"                        // '://'
  545. - * "("                          // login
  546. - *      "(?:"
  547. - *              "(.+?)"         // user (optional)
  548. - *              "(?::(.+))?"    // password (optional)
  549. - *              "@"
  550. - *      ")?"
  551. - *      "("                     // hostport
  552. - *              "(.+?)"         // host
  553. - *              "(?::([[:digit:]]+))?" // port (optional)
  554. - *      ")"
  555. + * "(.+?)"                             // scheme
  556. + * "://"                               // '://'
  557. + * "("                                 // login
  558. + *     "(?:"
  559. + *     "(.+?)"                         // user (optional)
  560. + *             "(?::(.+))?"            // password (optional)
  561. + *             "@"
  562. + *     ")?"
  563. + *     "("                             // hostport
  564. + *             "(.+?)"                 // host
  565. + *             "(?::([[:digit:]]+))?"  // port (optional)
  566. + *     ")"
  567.   * ")"
  568. - * "(?:/(.*?))?"                // path (optional)
  569. + * "(?:/(.*?))?"                       // path (optional)
  570. + * "(?:\?(.*?))?"                      // URL parameters (optional)
  571.   * "$"
  572.   */
  573.  
  574.  #define DBGNULLSTR(s) (((s)!=NULL)?(s):"<NULL>")
  575. -#if 0
  576. +#if 0 || defined(TEST_URLPARSER)
  577.  #define D(x) x
  578.  #else
  579.  #define D(x)
  580.  #endif
  581.  
  582.  static
  583. -void urldecodestr(char *dst, const char *src, size_t len)
  584. +void urldecodestr(char *outbuff, const char *buffer, size_t len)
  585.  {
  586. -       /*
  587. -        * Unicode characters with a code point > 255 are encoded
  588. -        * as UTF-8 bytes
  589. -        */
  590. -#define isurlxdigit(c) \
  591. -       (((c) >= '0' && (c) <= '9') || \
  592. -       ((c) >= 'a' && (c) <= 'f') || \
  593. -       ((c) >= 'A' && (c) <= 'F'))
  594. -       char a, b;
  595. -       while (*src && len--) {
  596. -               if (len > 2) {
  597. -                       if ((*src == '%') &&
  598. -                               (a = src[1]) && (b = src[2])) {
  599. -                               if ((isurlxdigit(a) &&
  600. -                                       isurlxdigit(b))) {
  601. -                                       if (a >= 'a')
  602. -                                               a -= 'a'-'A';
  603. -                                       if (a >= 'A')
  604. -                                               a -= ('A' - 10);
  605. -                                       else
  606. -                                               a -= '0';
  607. -
  608. -                                       if (b >= 'a')
  609. -                                               b -= 'a'-'A';
  610. -                                       if (b >= 'A')
  611. -                                               b -= ('A' - 10);
  612. -                                       else
  613. -                                               b -= '0';
  614. -
  615. -                                       *dst++ = 16*a+b;
  616. -
  617. -                                       src+=3;
  618. -                                       len-=2;
  619. -                                       continue;
  620. +       size_t i, j;
  621. +
  622. +       for (i = j = 0 ; i < len ; ) {
  623. +               switch (buffer[i]) {
  624. +                       case '%':
  625. +                               if ((i + 2) < len) {
  626. +                                       if (isxdigit((int)buffer[i+1]) && isxdigit((int)buffer[i+2])) {
  627. +                                               const char hexstr[3] = {
  628. +                                                       buffer[i+1],
  629. +                                                       buffer[i+2],
  630. +                                                       '\0'
  631. +                                               };
  632. +                                               outbuff[j++] = (unsigned char)strtol(hexstr, NULL, 16);
  633. +                                               i += 3;
  634. +                                       } else {
  635. +                                               /* invalid hex digit */
  636. +                                               outbuff[j++] = buffer[i];
  637. +                                               i++;
  638. +                                       }
  639. +                               } else {
  640. +                                       /* incomplete hex digit */
  641. +                                       outbuff[j++] = buffer[i];
  642. +                                       i++;
  643.                                 }
  644. -                       }
  645. -                }
  646. -               if (*src == '+') {
  647. -                       *dst++ = ' ';
  648. -                       src++;
  649. -                       continue;
  650. -                }
  651. -               *dst++ = *src++;
  652. +                               break;
  653. +                       case '+':
  654. +                               outbuff[j++] = ' ';
  655. +                               i++;
  656. +                               break;
  657. +                       default:
  658. +                               outbuff[j++] = buffer[i++];
  659. +                               break;
  660. +               }
  661.         }
  662. -       *dst++ = '\0';
  663. +
  664. +       outbuff[j] = '\0';
  665.  }
  666.  
  667.  url_parser_context *url_parser_create_context(const char *in_url, unsigned int flags)
  668.  {
  669. -       url_parser_context *uctx;
  670. +       url_parser_context_private *uctx;
  671.         char *s;
  672.         size_t in_url_len;
  673.         size_t context_len;
  674. @@ -137,30 +143,36 @@ url_parser_context *url_parser_create_context(const char *in_url, unsigned int f
  675.  
  676.         in_url_len = strlen(in_url);
  677.  
  678. -       context_len = sizeof(url_parser_context) +
  679. -               ((in_url_len+1)*5);
  680. +       context_len = sizeof(url_parser_context_private) +
  681. +               ((in_url_len+1)*6) +
  682. +               (sizeof(url_parser_name_value)*MAX_URL_PARAMETERS)+sizeof(void*);
  683.         uctx = malloc(context_len);
  684.         if (!uctx)
  685.                 return NULL;
  686.  
  687.         s = (void *)(uctx+1);
  688. -       uctx->in_url = s;               s+= in_url_len+1;
  689. -       (void)strcpy(uctx->in_url, in_url);
  690. -       uctx->scheme = s;               s+= in_url_len+1;
  691. -       uctx->login.username = s;       s+= in_url_len+1;
  692. -       uctx->hostport.hostname = s;    s+= in_url_len+1;
  693. -       uctx->path = s;                 s+= in_url_len+1;
  694. -       uctx->hostport.port = -1;
  695. -
  696. -       return uctx;
  697. +       uctx->c.in_url = s;             s+= in_url_len+1;
  698. +       (void)strcpy(uctx->c.in_url, in_url);
  699. +       uctx->c.scheme = s;             s+= in_url_len+1;
  700. +       uctx->c.login.username = s;     s+= in_url_len+1;
  701. +       uctx->c.hostport.hostname = s;  s+= in_url_len+1;
  702. +       uctx->c.path = s;               s+= in_url_len+1;
  703. +       uctx->c.hostport.port = -1;
  704. +       uctx->c.num_parameters = -1;
  705. +       uctx->c.parameters = (void *)s;         s+= (sizeof(url_parser_name_value)*MAX_URL_PARAMETERS)+sizeof(void*);
  706. +       uctx->parameter_string_buff = s;        s+= in_url_len+1;
  707. +
  708. +       return &uctx->c;
  709.  }
  710.  
  711. -int url_parser_parse(url_parser_context *uctx)
  712. +int url_parser_parse(url_parser_context *ctx)
  713.  {
  714. -       D((void)fprintf(stderr, "## parser in_url='%s'\n", uctx->in_url));
  715. +       url_parser_context_private *uctx = (url_parser_context_private *)ctx;
  716. +
  717. +       D((void)fprintf(stderr, "## parser in_url='%s'\n", uctx->c.in_url));
  718.  
  719.         char *s;
  720. -       const char *urlstr = uctx->in_url;
  721. +       const char *urlstr = uctx->c.in_url;
  722.         size_t slen;
  723.  
  724.         s = strstr(urlstr, "://");
  725. @@ -170,57 +182,120 @@ int url_parser_parse(url_parser_context *uctx)
  726.         }
  727.  
  728.         slen = s-urlstr;
  729. -       (void)memcpy(uctx->scheme, urlstr, slen);
  730. -       uctx->scheme[slen] = '\0';
  731. +       (void)memcpy(uctx->c.scheme, urlstr, slen);
  732. +       uctx->c.scheme[slen] = '\0';
  733.         urlstr += slen + 3;
  734.  
  735. -       D((void)fprintf(stdout, "scheme='%s', rest='%s'\n", uctx->scheme, urlstr));
  736. +       D((void)fprintf(stdout, "scheme='%s', rest='%s'\n", uctx->c.scheme, urlstr));
  737.  
  738.         s = strstr(urlstr, "@");
  739.         if (s) {
  740.                 /* URL has user/password */
  741.                 slen = s-urlstr;
  742. -               urldecodestr(uctx->login.username, urlstr, slen);
  743. +               urldecodestr(uctx->c.login.username, urlstr, slen);
  744.                 urlstr += slen + 1;
  745.  
  746. -               s = strstr(uctx->login.username, ":");
  747. +               s = strstr(uctx->c.login.username, ":");
  748.                 if (s) {
  749.                         /* found passwd */
  750. -                       uctx->login.passwd = s+1;
  751. +                       uctx->c.login.passwd = s+1;
  752.                         *s = '\0';
  753.                 }
  754.                 else
  755.                 {
  756. -                       uctx->login.passwd = NULL;
  757. +                       uctx->c.login.passwd = NULL;
  758.                 }
  759.  
  760.                 /* catch password-only URLs */
  761. -               if (uctx->login.username[0] == '\0')
  762. -                       uctx->login.username = NULL;
  763. +               if (uctx->c.login.username[0] == '\0')
  764. +                       uctx->c.login.username = NULL;
  765.         }
  766.         else
  767.         {
  768. -               uctx->login.username = NULL;
  769. -               uctx->login.passwd = NULL;
  770. +               uctx->c.login.username = NULL;
  771. +               uctx->c.login.passwd = NULL;
  772.         }
  773.  
  774.         D((void)fprintf(stdout, "login='%s', passwd='%s', rest='%s'\n",
  775. -               DBGNULLSTR(uctx->login.username),
  776. -               DBGNULLSTR(uctx->login.passwd),
  777. +               DBGNULLSTR(uctx->c.login.username),
  778. +               DBGNULLSTR(uctx->c.login.passwd),
  779.                 DBGNULLSTR(urlstr)));
  780.  
  781. +       char *raw_parameters;
  782. +
  783. +       uctx->c.num_parameters = 0;
  784. +       raw_parameters = strstr(urlstr, "?");
  785. +       if (raw_parameters) {
  786. +               *raw_parameters++ = '\0';
  787. +               D((void)fprintf(stdout, "raw parameters = '%s'\n", raw_parameters));
  788. +
  789. +               char *ps = raw_parameters;
  790. +               char *pv; /* parameter value */
  791. +               char *na; /* next '&' */
  792. +               char *pb = uctx->parameter_string_buff;
  793. +               char *pname;
  794. +               char *pvalue;
  795. +               ssize_t pi;
  796. +
  797. +               for (pi = 0; pi < MAX_URL_PARAMETERS ; pi++) {
  798. +                       pname = ps;
  799. +
  800. +                       /*
  801. +                        * Handle parameters without value,
  802. +                        * e.g. "path?name1&name2=value2"
  803. +                        */
  804. +                       na = strstr(ps, "&");
  805. +                       pv = strstr(ps, "=");
  806. +                       if (pv && (na?(na > pv):true)) {
  807. +                               *pv++ = '\0';
  808. +                               pvalue = pv;
  809. +                               ps = pv;
  810. +                       }
  811. +                       else {
  812. +                               pvalue = NULL;
  813. +                       }
  814. +
  815. +                       if (na) {
  816. +                               *na++ = '\0';
  817. +                       }
  818. +
  819. +                       /* URLDecode parameter name */
  820. +                       urldecodestr(pb, pname, strlen(pname));
  821. +                       uctx->c.parameters[pi].name = pb;
  822. +                       pb += strlen(uctx->c.parameters[pi].name)+1;
  823. +
  824. +                       /* URLDecode parameter value */
  825. +                       if (pvalue) {
  826. +                               urldecodestr(pb, pvalue, strlen(pvalue));
  827. +                               uctx->c.parameters[pi].value = pb;
  828. +                               pb += strlen(uctx->c.parameters[pi].value)+1;
  829. +                       }
  830. +                       else {
  831. +                               uctx->c.parameters[pi].value = NULL;
  832. +                       }
  833. +
  834. +                       /* Next '&' ? */
  835. +                       if (!na)
  836. +                               break;
  837. +
  838. +                       ps = na;
  839. +               }
  840. +
  841. +               uctx->c.num_parameters = pi+1;
  842. +       }
  843. +
  844.         s = strstr(urlstr, "/");
  845.         if (s) {
  846.                 /* URL has hostport */
  847.                 slen = s-urlstr;
  848. -               urldecodestr(uctx->hostport.hostname, urlstr, slen);
  849. +               urldecodestr(uctx->c.hostport.hostname, urlstr, slen);
  850.                 urlstr += slen + 1;
  851.  
  852.                 /*
  853.                  * check for addresses within '[' and ']', like
  854.                  * IPv6 addresses
  855.                  */
  856. -               s = uctx->hostport.hostname;
  857. +               s = uctx->c.hostport.hostname;
  858.                 if (s[0] == '[')
  859.                         s = strstr(s, "]");
  860.  
  861. @@ -232,29 +307,41 @@ int url_parser_parse(url_parser_context *uctx)
  862.                 s = strstr(s, ":");
  863.                 if (s) {
  864.                         /* found port number */
  865. -                       uctx->hostport.port = atoi(s+1);
  866. +                       uctx->c.hostport.port = atoi(s+1);
  867.                         *s = '\0';
  868.                 }
  869.         }
  870.         else
  871.         {
  872. -               (void)strcpy(uctx->hostport.hostname, urlstr);
  873. -               uctx->path = NULL;
  874. +               (void)strcpy(uctx->c.hostport.hostname, urlstr);
  875. +               uctx->c.path = NULL;
  876.                 urlstr = NULL;
  877.         }
  878.  
  879. -       D((void)fprintf(stdout, "hostport='%s', port=%d, rest='%s'\n",
  880. -               DBGNULLSTR(uctx->hostport.hostname),
  881. -               uctx->hostport.port,
  882. -               DBGNULLSTR(urlstr)));
  883. +       D((void)fprintf(stdout, "hostport='%s', port=%d, rest='%s', num_parameters=%d\n",
  884. +               DBGNULLSTR(uctx->c.hostport.hostname),
  885. +               uctx->c.hostport.port,
  886. +               DBGNULLSTR(urlstr),
  887. +               (int)uctx->c.num_parameters));
  888. +
  889. +       D(
  890. +               ssize_t dpi;
  891. +               for (dpi = 0 ; dpi < uctx->c.num_parameters ; dpi++) {
  892. +                       (void)fprintf(stdout, "param[%d]: name='%s'/value='%s'\n",
  893. +                               (int)dpi,
  894. +                               uctx->c.parameters[dpi].name,
  895. +                               DBGNULLSTR(uctx->c.parameters[dpi].value));
  896. +               }
  897. +       );
  898.  
  899.         if (!urlstr) {
  900. -               return 0;
  901. +               goto done;
  902.         }
  903.  
  904. -       urldecodestr(uctx->path, urlstr, strlen(urlstr));
  905. -       D((void)fprintf(stdout, "path='%s'\n", uctx->path));
  906. +       urldecodestr(uctx->c.path, urlstr, strlen(urlstr));
  907. +       D((void)fprintf(stdout, "path='%s'\n", uctx->c.path));
  908.  
  909. +done:
  910.         return 0;
  911.  }
  912.  
  913. @@ -301,7 +388,16 @@ int main(int ac, char *av[])
  914.         (void)test_url_parser("nfs://hostbar:93//absolutepath/a");
  915.         (void)test_url_parser("nfs://hostbar:93//absolutepath/blank%20path/a");
  916.         (void)test_url_parser("nfs://hostbar:93//absolutepath/blank+path/a");
  917. -
  918. +       (void)test_url_parser("foo://hostbar:93?param1");
  919. +       (void)test_url_parser("foo://hostbar:93?pname1=pvalue1");
  920. +       (void)test_url_parser("foo://hostbar:93?pname1=pvalue1&pname2=pvalue2");
  921. +       (void)test_url_parser("foo://hostbar:93?pname1=pvalue1&pvalue2=v2&n3=v3");
  922. +       (void)test_url_parser("foo://hostbar:93?pname1&param2=p2");
  923. +       (void)test_url_parser("foo://hostbar:93?pname1=&param2=p2");
  924. +       (void)test_url_parser("foo://hostbar:93//path/path2?param1=p1");
  925. +       (void)test_url_parser("foo://hostbar:93//path/path2?param1&param2=p2");
  926. +       (void)test_url_parser("foo://hostbar:93?pname1=pvalue1&%E2%82%AC=u+n2&n3=v3");
  927. +       (void)test_url_parser("foo://hostbar:93?pname1=pvalue1&%E2%82%AC=%E2%82%AC&n3=v3");
  928.  
  929.         (void)test_url_parser("foo://");
  930.         (void)test_url_parser("typo:/hostbar");
  931. diff --git a/mount/urlparser1.h b/mount/urlparser1.h
  932. index 8ed0f91..4526f1c 100644
  933. --- a/mount/urlparser1.h
  934. +++ b/mount/urlparser1.h
  935. @@ -26,8 +26,16 @@
  936.  
  937.  #include <stdlib.h>
  938.  
  939. -typedef
  940. -struct _url_parser_context {
  941. +#ifdef _MSC_VER
  942. +typedef signed long long ssize_t;
  943. +#endif
  944. +
  945. +typedef struct _url_parser_name_value {
  946. +       char *name;
  947. +       char *value;
  948. +} url_parser_name_value;
  949. +
  950. +typedef struct _url_parser_context {
  951.         char *in_url;
  952.  
  953.         char *scheme;
  954. @@ -40,6 +48,9 @@ struct _url_parser_context {
  955.                 signed int port;
  956.         } hostport;
  957.         char *path;
  958. +
  959. +       ssize_t num_parameters;
  960. +       url_parser_name_value *parameters;
  961.  } url_parser_context;
  962.  
  963.  /* Prototypes */
  964. --
  965. 2.45.1
  966.  
  967. From 8d4f28100296b12b5f673b33ff549ef6a68c5865 Mon Sep 17 00:00:00 2001
  968. From: Roland Mainz <roland.mainz@nrubsig.org>
  969. Date: Tue, 9 Jul 2024 15:40:15 +0200
  970. Subject: [PATCH 2/3] mount: Override options provided via URL parameters via
  971.  -o mount options
  972.  
  973. Override options provided via URL parameters via -o mount options,
  974. e.g. if nfs://foo.bar//path?ro=1 makes the mount read-only by
  975. default, then -o rw should override this.
  976.  
  977. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  978. ---
  979. mount/mount.c   | 80 ++++++++++++++++++++++++++++++++++++-------------
  980.  mount/options.c |  6 ++++
  981.  2 files changed, 65 insertions(+), 21 deletions(-)
  982.  
  983. diff --git a/mount/mount.c b/mount/mount.c
  984. index 06a0c35..1df510c 100644
  985. --- a/mount/mount.c
  986. +++ b/mount/mount.c
  987. @@ -50,17 +50,21 @@
  988.  DWORD EnumMounts(
  989.      IN LPNETRESOURCE pContainer);
  990.  
  991. +static DWORD ParseRemoteName(
  992. +    IN LPTSTR pRemoteName,
  993. +    IN OUT PMOUNT_OPTION_LIST pOptions,
  994. +    OUT LPTSTR pParsedRemoteName,
  995. +    OUT LPTSTR pConnectionName,
  996. +    IN size_t cchConnectionLen);
  997.  static DWORD DoMount(
  998.      IN LPTSTR pLocalName,
  999.      IN LPTSTR pRemoteName,
  1000. +    IN LPTSTR pParsedRemoteName,
  1001.      IN BOOL bPersistent,
  1002.      IN PMOUNT_OPTION_LIST pOptions);
  1003.  static DWORD DoUnmount(
  1004.      IN LPTSTR pLocalName,
  1005.      IN BOOL bForce);
  1006. -
  1007. -static void RecursivePrintEaInformation(
  1008. -    IN PFILE_FULL_EA_INFORMATION EA);
  1009.  static BOOL ParseDriveLetter(
  1010.      IN LPTSTR pArg,
  1011.      OUT PTCH pDriveLetter);
  1012. @@ -142,6 +146,9 @@ DWORD __cdecl _tmain(DWORD argc, LPTSTR argv[])
  1013.      BOOL    bForceUnmount = FALSE;
  1014.      BOOL    bPersistent = FALSE;
  1015.      MOUNT_OPTION_LIST Options;
  1016. +#define MAX_MNTOPTS 128
  1017. +    TCHAR   *mntopts[MAX_MNTOPTS] = { 0 };
  1018. +    size_t  num_mntopts = 0;
  1019.  
  1020.      int crtsetdbgflags = 0;
  1021.      crtsetdbgflags |= _CRTDBG_ALLOC_MEM_DF;  /* use debug heap */
  1022. @@ -201,11 +208,14 @@ DWORD __cdecl _tmain(DWORD argc, LPTSTR argv[])
  1023.                      goto out_free;
  1024.                  }
  1025.  
  1026. -                if (!ParseMountOptions(argv[i], &Options))
  1027. -                {
  1028. +                if (num_mntopts >= (MAX_MNTOPTS-1)) {
  1029.                      result = ERROR_BAD_ARGUMENTS;
  1030. +                    _ftprintf(stderr, TEXT("Too many -o ")
  1031. +                        TEXT("options.\n\n"));
  1032.                      goto out_free;
  1033.                  }
  1034. +
  1035. +                mntopts[num_mntopts++] = argv[i];
  1036.              }
  1037.             /*
  1038.              * Filesystem type, we use this for Solaris
  1039. @@ -282,6 +292,11 @@ DWORD __cdecl _tmain(DWORD argc, LPTSTR argv[])
  1040.      }
  1041.      else /* mount */
  1042.      {
  1043. +        TCHAR szRemoteName[NFS41_SYS_MAX_PATH_LEN];
  1044. +        TCHAR szParsedRemoteName[NFS41_SYS_MAX_PATH_LEN];
  1045. +
  1046. +        *szRemoteName = TEXT('\0');
  1047. +
  1048.          if (pRemoteName == NULL)
  1049.          {
  1050.              result = ERROR_BAD_NET_NAME;
  1051. @@ -290,7 +305,29 @@ DWORD __cdecl _tmain(DWORD argc, LPTSTR argv[])
  1052.              goto out_free;
  1053.          }
  1054.  
  1055. -        result = DoMount(szLocalName, pRemoteName, bPersistent, &Options);
  1056. +        /*
  1057. +         * First we need to parse the remote name, which might be a
  1058. +         * nfs://-URL with URL parameters, which provide default
  1059. +         * options for a NFS mount point, which can be overridden via
  1060. +         * -o below.
  1061. +         */
  1062. +        result = ParseRemoteName(pRemoteName, &Options,
  1063. +            szParsedRemoteName, szRemoteName, NFS41_SYS_MAX_PATH_LEN);
  1064. +        if (result)
  1065. +            goto out;
  1066. +
  1067. +        /*
  1068. +         * Parse saved -o options (possibly overriding defaults
  1069. +         * provided via (nfs://-)URL parameters above.
  1070. +         */
  1071. +        for (i = 0 ; i < num_mntopts ; i++) {
  1072. +            if (!ParseMountOptions(mntopts[i], &Options)) {
  1073. +                result = ERROR_BAD_ARGUMENTS;
  1074. +                goto out_free;
  1075. +            }
  1076. +        }
  1077. +
  1078. +        result = DoMount(szLocalName, szRemoteName, szParsedRemoteName, bPersistent, &Options);
  1079.          if (result)
  1080.              PrintErrorMessage(result);
  1081.      }
  1082. @@ -682,19 +719,25 @@ out:
  1083.  static DWORD DoMount(
  1084.      IN LPTSTR pLocalName,
  1085.      IN LPTSTR pRemoteName,
  1086. +    IN LPTSTR pParsedRemoteName,
  1087.      IN BOOL bPersistent,
  1088.      IN PMOUNT_OPTION_LIST pOptions)
  1089.  {
  1090.      DWORD result = NO_ERROR;
  1091.      TCHAR szExisting[NFS41_SYS_MAX_PATH_LEN];
  1092. -    TCHAR szParsedRemoteName[NFS41_SYS_MAX_PATH_LEN];
  1093. -    TCHAR szRemoteName[NFS41_SYS_MAX_PATH_LEN];
  1094.      DWORD dwLength;
  1095. +    NETRESOURCE NetResource;
  1096.  
  1097. -    *szRemoteName = TEXT('\0');
  1098. -    result = ParseRemoteName(pRemoteName, pOptions, szParsedRemoteName, szRemoteName, NFS41_SYS_MAX_PATH_LEN);
  1099. -    if (result)
  1100. -        goto out;
  1101. +
  1102. +    if (pOptions->Buffer->Length) {
  1103. +        if (pOptions->Current)
  1104. +            pOptions->Current->NextEntryOffset = 0;
  1105. +        NetResource.lpComment = (LPTSTR)&pOptions->Buffer[0];
  1106. +    }
  1107. +
  1108. +#if 0
  1109. +    RecursivePrintEaInformation((PFILE_FULL_EA_INFORMATION)pOptions->Buffer->Buffer);
  1110. +#endif
  1111.  
  1112.      /* fail if the connection already exists */
  1113.      dwLength = NFS41_SYS_MAX_PATH_LEN;
  1114. @@ -708,7 +751,6 @@ static DWORD DoMount(
  1115.      }
  1116.      else
  1117.      {
  1118. -        NETRESOURCE NetResource;
  1119.          TCHAR szConnection[NFS41_SYS_MAX_PATH_LEN];
  1120.          DWORD ConnectSize = NFS41_SYS_MAX_PATH_LEN, ConnectResult, Flags = 0;
  1121.  
  1122. @@ -716,14 +758,11 @@ static DWORD DoMount(
  1123.          NetResource.dwType = RESOURCETYPE_DISK;
  1124.          /* drive letter is chosen automatically if lpLocalName == NULL */
  1125.          NetResource.lpLocalName = *pLocalName == TEXT('*') ? NULL : pLocalName;
  1126. -        NetResource.lpRemoteName = szRemoteName;
  1127. +        NetResource.lpRemoteName = pRemoteName;
  1128.          /* ignore other network providers */
  1129.          NetResource.lpProvider = TEXT(NFS41_PROVIDER_NAME_A);
  1130.          /* pass mount options via lpComment */
  1131. -        if (pOptions->Buffer->Length)
  1132. -        {
  1133. -            if (pOptions->Current)
  1134. -                pOptions->Current->NextEntryOffset = 0;
  1135. +        if (pOptions->Buffer->Length) {
  1136.              NetResource.lpComment = (LPTSTR)pOptions->Buffer;
  1137.          }
  1138.  
  1139. @@ -736,14 +775,13 @@ static DWORD DoMount(
  1140.  
  1141.          if (result == NO_ERROR)
  1142.              _tprintf(TEXT("Successfully mounted '%s' to drive '%s'\n"),
  1143. -                szParsedRemoteName, szConnection);
  1144. +                pParsedRemoteName, szConnection);
  1145.          else
  1146.              _ftprintf(stderr, TEXT("WNetUseConnection(%s, %s) ")
  1147.                  TEXT("failed with error code %u.\n"),
  1148. -                pLocalName, szRemoteName, result);
  1149. +                pLocalName, pRemoteName, result);
  1150.      }
  1151.  
  1152. -out:
  1153.      return result;
  1154.  }
  1155.  
  1156. diff --git a/mount/options.c b/mount/options.c
  1157. index 4be4e8b..b004302 100644
  1158. --- a/mount/options.c
  1159. +++ b/mount/options.c
  1160. @@ -108,6 +108,11 @@ BOOL InsertOption(
  1161.      USHORT ValueLen = (USHORT)_tcslen(Value) * sizeof(TCHAR);
  1162.      ULONG SpaceRequired = EaBufferSize(NameLen, ValueLen);
  1163.  
  1164. +    /*
  1165. +     * FIXME: Some duplicates are wanted, e.g. "rw" overriding "ro" etc
  1166. +     * So better just let the kernel do the work
  1167. +     */
  1168. +#if 0
  1169.      /* don't allow duplicate options */
  1170.      if (FindOptionByName(Name, Options, &Current)) {
  1171.          _ftprintf(stderr, TEXT("Found a duplicate option ")
  1172. @@ -118,6 +123,7 @@ BOOL InsertOption(
  1173.              Name, ValueLen ? TEXT("=") : Value, Value);
  1174.          return FALSE;
  1175.      }
  1176. +#endif
  1177.  
  1178.      /* fail if we're out of space */
  1179.      if (SpaceRequired > Options->Remaining) {
  1180. --
  1181. 2.45.1
  1182.  
  1183. From e491025e5de16b5fd9d30477e06cfabdb4eb3196 Mon Sep 17 00:00:00 2001
  1184. From: Roland Mainz <roland.mainz@nrubsig.org>
  1185. Date: Tue, 9 Jul 2024 15:51:25 +0200
  1186. Subject: [PATCH 3/3] mount: Add SysV+Linux -r and Linux -w options
  1187.  
  1188. Add SysV+Linux -r (alias for "-o ro") and Linux -w (alias for "-o rw")
  1189. options.
  1190.  
  1191. Reported-by: Josh Hurst <joshhurst@gmail.com>
  1192. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  1193. ---
  1194. mount/mount.c | 22 ++++++++++++++++++++++
  1195.  1 file changed, 22 insertions(+)
  1196.  
  1197. diff --git a/mount/mount.c b/mount/mount.c
  1198. index 1df510c..a53228a 100644
  1199. --- a/mount/mount.c
  1200. +++ b/mount/mount.c
  1201. @@ -87,6 +87,8 @@ static VOID PrintUsage(LPTSTR pProcess)
  1202.             " (Linux compat)\n")
  1203.          TEXT("\t-p\tmake the mount persist over reboots\n")
  1204.          TEXT("\t-o <comma-separated mount options>\n")
  1205. +        TEXT("\t-r\tAlias for -o ro (read-only mount)\n")
  1206. +        TEXT("\t-w\tAlias for -o rw (read-write mount)\n")
  1207.  
  1208.          TEXT("* Mount options:\n")
  1209.          TEXT("\tro\tmount as read-only\n")
  1210. @@ -217,6 +219,26 @@ DWORD __cdecl _tmain(DWORD argc, LPTSTR argv[])
  1211.  
  1212.                  mntopts[num_mntopts++] = argv[i];
  1213.              }
  1214. +            else if (_tcscmp(argv[i], TEXT("-r")) == 0) /* mount option */
  1215. +            {
  1216. +                if (num_mntopts >= (MAX_MNTOPTS-1)) {
  1217. +                    result = ERROR_BAD_ARGUMENTS;
  1218. +                    (void)_ftprintf(stderr, TEXT("Too many options.\n\n"));
  1219. +                    goto out_free;
  1220. +                }
  1221. +
  1222. +                mntopts[num_mntopts++] = TEXT("ro");
  1223. +            }
  1224. +            else if (_tcscmp(argv[i], TEXT("-w")) == 0) /* mount option */
  1225. +            {
  1226. +                if (num_mntopts >= (MAX_MNTOPTS-1)) {
  1227. +                    result = ERROR_BAD_ARGUMENTS;
  1228. +                    (void)_ftprintf(stderr, TEXT("Too many options.\n\n"));
  1229. +                    goto out_free;
  1230. +                }
  1231. +
  1232. +                mntopts[num_mntopts++] = TEXT("rw");
  1233. +            }
  1234.             /*
  1235.              * Filesystem type, we use this for Solaris
  1236.              * $ mount(1M) -F nfs ... # and Linux
  1237. --
  1238. 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