ncplib_kernel.c 32 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312
  1. /*
  2. * ncplib_kernel.c
  3. *
  4. * Copyright (C) 1995, 1996 by Volker Lendecke
  5. * Modified for big endian by J.F. Chadima and David S. Miller
  6. * Modified 1997 Peter Waltenberg, Bill Hawes, David Woodhouse for 2.1 dcache
  7. * Modified 1999 Wolfram Pienkoss for NLS
  8. * Modified 2000 Ben Harris, University of Cambridge for NFS NS meta-info
  9. *
  10. */
  11. #include "ncplib_kernel.h"
  12. static inline void assert_server_locked(struct ncp_server *server)
  13. {
  14. if (server->lock == 0) {
  15. DPRINTK("ncpfs: server not locked!\n");
  16. }
  17. }
  18. static void ncp_add_byte(struct ncp_server *server, __u8 x)
  19. {
  20. assert_server_locked(server);
  21. *(__u8 *) (&(server->packet[server->current_size])) = x;
  22. server->current_size += 1;
  23. return;
  24. }
  25. static void ncp_add_word(struct ncp_server *server, __le16 x)
  26. {
  27. assert_server_locked(server);
  28. put_unaligned(x, (__le16 *) (&(server->packet[server->current_size])));
  29. server->current_size += 2;
  30. return;
  31. }
  32. static void ncp_add_be16(struct ncp_server *server, __u16 x)
  33. {
  34. assert_server_locked(server);
  35. put_unaligned(cpu_to_be16(x), (__be16 *) (&(server->packet[server->current_size])));
  36. server->current_size += 2;
  37. }
  38. static void ncp_add_dword(struct ncp_server *server, __le32 x)
  39. {
  40. assert_server_locked(server);
  41. put_unaligned(x, (__le32 *) (&(server->packet[server->current_size])));
  42. server->current_size += 4;
  43. return;
  44. }
  45. static void ncp_add_be32(struct ncp_server *server, __u32 x)
  46. {
  47. assert_server_locked(server);
  48. put_unaligned(cpu_to_be32(x), (__be32 *)(&(server->packet[server->current_size])));
  49. server->current_size += 4;
  50. }
  51. static inline void ncp_add_dword_lh(struct ncp_server *server, __u32 x) {
  52. ncp_add_dword(server, cpu_to_le32(x));
  53. }
  54. static void ncp_add_mem(struct ncp_server *server, const void *source, int size)
  55. {
  56. assert_server_locked(server);
  57. memcpy(&(server->packet[server->current_size]), source, size);
  58. server->current_size += size;
  59. return;
  60. }
  61. static void ncp_add_pstring(struct ncp_server *server, const char *s)
  62. {
  63. int len = strlen(s);
  64. assert_server_locked(server);
  65. if (len > 255) {
  66. DPRINTK("ncpfs: string too long: %s\n", s);
  67. len = 255;
  68. }
  69. ncp_add_byte(server, len);
  70. ncp_add_mem(server, s, len);
  71. return;
  72. }
  73. static inline void ncp_init_request(struct ncp_server *server)
  74. {
  75. ncp_lock_server(server);
  76. server->current_size = sizeof(struct ncp_request_header);
  77. server->has_subfunction = 0;
  78. }
  79. static inline void ncp_init_request_s(struct ncp_server *server, int subfunction)
  80. {
  81. ncp_lock_server(server);
  82. server->current_size = sizeof(struct ncp_request_header) + 2;
  83. ncp_add_byte(server, subfunction);
  84. server->has_subfunction = 1;
  85. }
  86. static inline char *
  87. ncp_reply_data(struct ncp_server *server, int offset)
  88. {
  89. return &(server->packet[sizeof(struct ncp_reply_header) + offset]);
  90. }
  91. static inline u8 BVAL(void *data)
  92. {
  93. return *(u8 *)data;
  94. }
  95. static u8 ncp_reply_byte(struct ncp_server *server, int offset)
  96. {
  97. return *(u8 *)ncp_reply_data(server, offset);
  98. }
  99. static inline u16 WVAL_LH(void *data)
  100. {
  101. return get_unaligned_le16(data);
  102. }
  103. static u16
  104. ncp_reply_le16(struct ncp_server *server, int offset)
  105. {
  106. return get_unaligned_le16(ncp_reply_data(server, offset));
  107. }
  108. static u16
  109. ncp_reply_be16(struct ncp_server *server, int offset)
  110. {
  111. return get_unaligned_be16(ncp_reply_data(server, offset));
  112. }
  113. static inline u32 DVAL_LH(void *data)
  114. {
  115. return get_unaligned_le32(data);
  116. }
  117. static __le32
  118. ncp_reply_dword(struct ncp_server *server, int offset)
  119. {
  120. return get_unaligned((__le32 *)ncp_reply_data(server, offset));
  121. }
  122. static inline __u32 ncp_reply_dword_lh(struct ncp_server* server, int offset) {
  123. return le32_to_cpu(ncp_reply_dword(server, offset));
  124. }
  125. int
  126. ncp_negotiate_buffersize(struct ncp_server *server, int size, int *target)
  127. {
  128. int result;
  129. ncp_init_request(server);
  130. ncp_add_be16(server, size);
  131. if ((result = ncp_request(server, 33)) != 0) {
  132. ncp_unlock_server(server);
  133. return result;
  134. }
  135. *target = min_t(unsigned int, ncp_reply_be16(server, 0), size);
  136. ncp_unlock_server(server);
  137. return 0;
  138. }
  139. /* options:
  140. * bit 0 ipx checksum
  141. * bit 1 packet signing
  142. */
  143. int
  144. ncp_negotiate_size_and_options(struct ncp_server *server,
  145. int size, int options, int *ret_size, int *ret_options) {
  146. int result;
  147. /* there is minimum */
  148. if (size < NCP_BLOCK_SIZE) size = NCP_BLOCK_SIZE;
  149. ncp_init_request(server);
  150. ncp_add_be16(server, size);
  151. ncp_add_byte(server, options);
  152. if ((result = ncp_request(server, 0x61)) != 0)
  153. {
  154. ncp_unlock_server(server);
  155. return result;
  156. }
  157. /* NCP over UDP returns 0 (!!!) */
  158. result = ncp_reply_be16(server, 0);
  159. if (result >= NCP_BLOCK_SIZE)
  160. size = min(result, size);
  161. *ret_size = size;
  162. *ret_options = ncp_reply_byte(server, 4);
  163. ncp_unlock_server(server);
  164. return 0;
  165. }
  166. int ncp_get_volume_info_with_number(struct ncp_server* server,
  167. int n, struct ncp_volume_info* target) {
  168. int result;
  169. int len;
  170. ncp_init_request_s(server, 44);
  171. ncp_add_byte(server, n);
  172. if ((result = ncp_request(server, 22)) != 0) {
  173. goto out;
  174. }
  175. target->total_blocks = ncp_reply_dword_lh(server, 0);
  176. target->free_blocks = ncp_reply_dword_lh(server, 4);
  177. target->purgeable_blocks = ncp_reply_dword_lh(server, 8);
  178. target->not_yet_purgeable_blocks = ncp_reply_dword_lh(server, 12);
  179. target->total_dir_entries = ncp_reply_dword_lh(server, 16);
  180. target->available_dir_entries = ncp_reply_dword_lh(server, 20);
  181. target->sectors_per_block = ncp_reply_byte(server, 28);
  182. memset(&(target->volume_name), 0, sizeof(target->volume_name));
  183. result = -EIO;
  184. len = ncp_reply_byte(server, 29);
  185. if (len > NCP_VOLNAME_LEN) {
  186. DPRINTK("ncpfs: volume name too long: %d\n", len);
  187. goto out;
  188. }
  189. memcpy(&(target->volume_name), ncp_reply_data(server, 30), len);
  190. result = 0;
  191. out:
  192. ncp_unlock_server(server);
  193. return result;
  194. }
  195. int ncp_get_directory_info(struct ncp_server* server, __u8 n,
  196. struct ncp_volume_info* target) {
  197. int result;
  198. int len;
  199. ncp_init_request_s(server, 45);
  200. ncp_add_byte(server, n);
  201. if ((result = ncp_request(server, 22)) != 0) {
  202. goto out;
  203. }
  204. target->total_blocks = ncp_reply_dword_lh(server, 0);
  205. target->free_blocks = ncp_reply_dword_lh(server, 4);
  206. target->purgeable_blocks = 0;
  207. target->not_yet_purgeable_blocks = 0;
  208. target->total_dir_entries = ncp_reply_dword_lh(server, 8);
  209. target->available_dir_entries = ncp_reply_dword_lh(server, 12);
  210. target->sectors_per_block = ncp_reply_byte(server, 20);
  211. memset(&(target->volume_name), 0, sizeof(target->volume_name));
  212. result = -EIO;
  213. len = ncp_reply_byte(server, 21);
  214. if (len > NCP_VOLNAME_LEN) {
  215. DPRINTK("ncpfs: volume name too long: %d\n", len);
  216. goto out;
  217. }
  218. memcpy(&(target->volume_name), ncp_reply_data(server, 22), len);
  219. result = 0;
  220. out:
  221. ncp_unlock_server(server);
  222. return result;
  223. }
  224. int
  225. ncp_close_file(struct ncp_server *server, const char *file_id)
  226. {
  227. int result;
  228. ncp_init_request(server);
  229. ncp_add_byte(server, 0);
  230. ncp_add_mem(server, file_id, 6);
  231. result = ncp_request(server, 66);
  232. ncp_unlock_server(server);
  233. return result;
  234. }
  235. int
  236. ncp_make_closed(struct inode *inode)
  237. {
  238. int err;
  239. err = 0;
  240. mutex_lock(&NCP_FINFO(inode)->open_mutex);
  241. if (atomic_read(&NCP_FINFO(inode)->opened) == 1) {
  242. atomic_set(&NCP_FINFO(inode)->opened, 0);
  243. err = ncp_close_file(NCP_SERVER(inode), NCP_FINFO(inode)->file_handle);
  244. if (!err)
  245. PPRINTK("ncp_make_closed: volnum=%d, dirent=%u, error=%d\n",
  246. NCP_FINFO(inode)->volNumber,
  247. NCP_FINFO(inode)->dirEntNum, err);
  248. }
  249. mutex_unlock(&NCP_FINFO(inode)->open_mutex);
  250. return err;
  251. }
  252. static void ncp_add_handle_path(struct ncp_server *server, __u8 vol_num,
  253. __le32 dir_base, int have_dir_base,
  254. const char *path)
  255. {
  256. ncp_add_byte(server, vol_num);
  257. ncp_add_dword(server, dir_base);
  258. if (have_dir_base != 0) {
  259. ncp_add_byte(server, 1); /* dir_base */
  260. } else {
  261. ncp_add_byte(server, 0xff); /* no handle */
  262. }
  263. if (path != NULL) {
  264. ncp_add_byte(server, 1); /* 1 component */
  265. ncp_add_pstring(server, path);
  266. } else {
  267. ncp_add_byte(server, 0);
  268. }
  269. }
  270. int ncp_dirhandle_alloc(struct ncp_server* server, __u8 volnum, __le32 dirent,
  271. __u8* dirhandle) {
  272. int result;
  273. ncp_init_request(server);
  274. ncp_add_byte(server, 12); /* subfunction */
  275. ncp_add_byte(server, NW_NS_DOS);
  276. ncp_add_byte(server, 0);
  277. ncp_add_word(server, 0);
  278. ncp_add_handle_path(server, volnum, dirent, 1, NULL);
  279. if ((result = ncp_request(server, 87)) == 0) {
  280. *dirhandle = ncp_reply_byte(server, 0);
  281. }
  282. ncp_unlock_server(server);
  283. return result;
  284. }
  285. int ncp_dirhandle_free(struct ncp_server* server, __u8 dirhandle) {
  286. int result;
  287. ncp_init_request_s(server, 20);
  288. ncp_add_byte(server, dirhandle);
  289. result = ncp_request(server, 22);
  290. ncp_unlock_server(server);
  291. return result;
  292. }
  293. void ncp_extract_file_info(void *structure, struct nw_info_struct *target)
  294. {
  295. __u8 *name_len;
  296. const int info_struct_size = offsetof(struct nw_info_struct, nameLen);
  297. memcpy(target, structure, info_struct_size);
  298. name_len = structure + info_struct_size;
  299. target->nameLen = *name_len;
  300. memcpy(target->entryName, name_len + 1, *name_len);
  301. target->entryName[*name_len] = '\0';
  302. target->volNumber = le32_to_cpu(target->volNumber);
  303. return;
  304. }
  305. #ifdef CONFIG_NCPFS_NFS_NS
  306. static inline void ncp_extract_nfs_info(unsigned char *structure,
  307. struct nw_nfs_info *target)
  308. {
  309. target->mode = DVAL_LH(structure);
  310. target->rdev = DVAL_LH(structure + 8);
  311. }
  312. #endif
  313. int ncp_obtain_nfs_info(struct ncp_server *server,
  314. struct nw_info_struct *target)
  315. {
  316. int result = 0;
  317. #ifdef CONFIG_NCPFS_NFS_NS
  318. __u32 volnum = target->volNumber;
  319. if (ncp_is_nfs_extras(server, volnum)) {
  320. ncp_init_request(server);
  321. ncp_add_byte(server, 19); /* subfunction */
  322. ncp_add_byte(server, server->name_space[volnum]);
  323. ncp_add_byte(server, NW_NS_NFS);
  324. ncp_add_byte(server, 0);
  325. ncp_add_byte(server, volnum);
  326. ncp_add_dword(server, target->dirEntNum);
  327. /* We must retrieve both nlinks and rdev, otherwise some server versions
  328. report zeroes instead of valid data */
  329. ncp_add_dword_lh(server, NSIBM_NFS_MODE | NSIBM_NFS_NLINKS | NSIBM_NFS_RDEV);
  330. if ((result = ncp_request(server, 87)) == 0) {
  331. ncp_extract_nfs_info(ncp_reply_data(server, 0), &target->nfs);
  332. DPRINTK(KERN_DEBUG
  333. "ncp_obtain_nfs_info: (%s) mode=0%o, rdev=0x%x\n",
  334. target->entryName, target->nfs.mode,
  335. target->nfs.rdev);
  336. } else {
  337. target->nfs.mode = 0;
  338. target->nfs.rdev = 0;
  339. }
  340. ncp_unlock_server(server);
  341. } else
  342. #endif
  343. {
  344. target->nfs.mode = 0;
  345. target->nfs.rdev = 0;
  346. }
  347. return result;
  348. }
  349. /*
  350. * Returns information for a (one-component) name relative to
  351. * the specified directory.
  352. */
  353. int ncp_obtain_info(struct ncp_server *server, struct inode *dir, char *path,
  354. struct nw_info_struct *target)
  355. {
  356. __u8 volnum = NCP_FINFO(dir)->volNumber;
  357. __le32 dirent = NCP_FINFO(dir)->dirEntNum;
  358. int result;
  359. if (target == NULL) {
  360. printk(KERN_ERR "ncp_obtain_info: invalid call\n");
  361. return -EINVAL;
  362. }
  363. ncp_init_request(server);
  364. ncp_add_byte(server, 6); /* subfunction */
  365. ncp_add_byte(server, server->name_space[volnum]);
  366. ncp_add_byte(server, server->name_space[volnum]); /* N.B. twice ?? */
  367. ncp_add_word(server, cpu_to_le16(0x8006)); /* get all */
  368. ncp_add_dword(server, RIM_ALL);
  369. ncp_add_handle_path(server, volnum, dirent, 1, path);
  370. if ((result = ncp_request(server, 87)) != 0)
  371. goto out;
  372. ncp_extract_file_info(ncp_reply_data(server, 0), target);
  373. ncp_unlock_server(server);
  374. result = ncp_obtain_nfs_info(server, target);
  375. return result;
  376. out:
  377. ncp_unlock_server(server);
  378. return result;
  379. }
  380. #ifdef CONFIG_NCPFS_NFS_NS
  381. static int
  382. ncp_obtain_DOS_dir_base(struct ncp_server *server,
  383. __u8 volnum, __le32 dirent,
  384. char *path, /* At most 1 component */
  385. __le32 *DOS_dir_base)
  386. {
  387. int result;
  388. ncp_init_request(server);
  389. ncp_add_byte(server, 6); /* subfunction */
  390. ncp_add_byte(server, server->name_space[volnum]);
  391. ncp_add_byte(server, server->name_space[volnum]);
  392. ncp_add_word(server, cpu_to_le16(0x8006)); /* get all */
  393. ncp_add_dword(server, RIM_DIRECTORY);
  394. ncp_add_handle_path(server, volnum, dirent, 1, path);
  395. if ((result = ncp_request(server, 87)) == 0)
  396. {
  397. if (DOS_dir_base) *DOS_dir_base=ncp_reply_dword(server, 0x34);
  398. }
  399. ncp_unlock_server(server);
  400. return result;
  401. }
  402. #endif /* CONFIG_NCPFS_NFS_NS */
  403. static inline int
  404. ncp_get_known_namespace(struct ncp_server *server, __u8 volume)
  405. {
  406. #if defined(CONFIG_NCPFS_OS2_NS) || defined(CONFIG_NCPFS_NFS_NS)
  407. int result;
  408. __u8 *namespace;
  409. __u16 no_namespaces;
  410. ncp_init_request(server);
  411. ncp_add_byte(server, 24); /* Subfunction: Get Name Spaces Loaded */
  412. ncp_add_word(server, 0);
  413. ncp_add_byte(server, volume);
  414. if ((result = ncp_request(server, 87)) != 0) {
  415. ncp_unlock_server(server);
  416. return NW_NS_DOS; /* not result ?? */
  417. }
  418. result = NW_NS_DOS;
  419. no_namespaces = ncp_reply_le16(server, 0);
  420. namespace = ncp_reply_data(server, 2);
  421. while (no_namespaces > 0) {
  422. DPRINTK("get_namespaces: found %d on %d\n", *namespace, volume);
  423. #ifdef CONFIG_NCPFS_NFS_NS
  424. if ((*namespace == NW_NS_NFS) && !(server->m.flags&NCP_MOUNT_NO_NFS))
  425. {
  426. result = NW_NS_NFS;
  427. break;
  428. }
  429. #endif /* CONFIG_NCPFS_NFS_NS */
  430. #ifdef CONFIG_NCPFS_OS2_NS
  431. if ((*namespace == NW_NS_OS2) && !(server->m.flags&NCP_MOUNT_NO_OS2))
  432. {
  433. result = NW_NS_OS2;
  434. }
  435. #endif /* CONFIG_NCPFS_OS2_NS */
  436. namespace += 1;
  437. no_namespaces -= 1;
  438. }
  439. ncp_unlock_server(server);
  440. return result;
  441. #else /* neither OS2 nor NFS - only DOS */
  442. return NW_NS_DOS;
  443. #endif /* defined(CONFIG_NCPFS_OS2_NS) || defined(CONFIG_NCPFS_NFS_NS) */
  444. }
  445. static int
  446. ncp_ObtainSpecificDirBase(struct ncp_server *server,
  447. __u8 nsSrc, __u8 nsDst, __u8 vol_num, __le32 dir_base,
  448. char *path, /* At most 1 component */
  449. __le32 *dirEntNum, __le32 *DosDirNum)
  450. {
  451. int result;
  452. ncp_init_request(server);
  453. ncp_add_byte(server, 6); /* subfunction */
  454. ncp_add_byte(server, nsSrc);
  455. ncp_add_byte(server, nsDst);
  456. ncp_add_word(server, cpu_to_le16(0x8006)); /* get all */
  457. ncp_add_dword(server, RIM_ALL);
  458. ncp_add_handle_path(server, vol_num, dir_base, 1, path);
  459. if ((result = ncp_request(server, 87)) != 0)
  460. {
  461. ncp_unlock_server(server);
  462. return result;
  463. }
  464. if (dirEntNum)
  465. *dirEntNum = ncp_reply_dword(server, 0x30);
  466. if (DosDirNum)
  467. *DosDirNum = ncp_reply_dword(server, 0x34);
  468. ncp_unlock_server(server);
  469. return 0;
  470. }
  471. int
  472. ncp_mount_subdir(struct ncp_server *server,
  473. __u8 volNumber, __u8 srcNS, __le32 dirEntNum,
  474. __u32* volume, __le32* newDirEnt, __le32* newDosEnt)
  475. {
  476. int dstNS;
  477. int result;
  478. dstNS = ncp_get_known_namespace(server, volNumber);
  479. if ((result = ncp_ObtainSpecificDirBase(server, srcNS, dstNS, volNumber,
  480. dirEntNum, NULL, newDirEnt, newDosEnt)) != 0)
  481. {
  482. return result;
  483. }
  484. server->name_space[volNumber] = dstNS;
  485. *volume = volNumber;
  486. server->m.mounted_vol[1] = 0;
  487. server->m.mounted_vol[0] = 'X';
  488. return 0;
  489. }
  490. int
  491. ncp_get_volume_root(struct ncp_server *server, const char *volname,
  492. __u32* volume, __le32* dirent, __le32* dosdirent)
  493. {
  494. int result;
  495. __u8 volnum;
  496. DPRINTK("ncp_get_volume_root: looking up vol %s\n", volname);
  497. ncp_init_request(server);
  498. ncp_add_byte(server, 22); /* Subfunction: Generate dir handle */
  499. ncp_add_byte(server, 0); /* DOS namespace */
  500. ncp_add_byte(server, 0); /* reserved */
  501. ncp_add_byte(server, 0); /* reserved */
  502. ncp_add_byte(server, 0); /* reserved */
  503. ncp_add_byte(server, 0); /* faked volume number */
  504. ncp_add_dword(server, 0); /* faked dir_base */
  505. ncp_add_byte(server, 0xff); /* Don't have a dir_base */
  506. ncp_add_byte(server, 1); /* 1 path component */
  507. ncp_add_pstring(server, volname);
  508. if ((result = ncp_request(server, 87)) != 0) {
  509. ncp_unlock_server(server);
  510. return result;
  511. }
  512. *dirent = *dosdirent = ncp_reply_dword(server, 4);
  513. volnum = ncp_reply_byte(server, 8);
  514. ncp_unlock_server(server);
  515. *volume = volnum;
  516. server->name_space[volnum] = ncp_get_known_namespace(server, volnum);
  517. DPRINTK("lookup_vol: namespace[%d] = %d\n",
  518. volnum, server->name_space[volnum]);
  519. return 0;
  520. }
  521. int
  522. ncp_lookup_volume(struct ncp_server *server, const char *volname,
  523. struct nw_info_struct *target)
  524. {
  525. int result;
  526. memset(target, 0, sizeof(*target));
  527. result = ncp_get_volume_root(server, volname,
  528. &target->volNumber, &target->dirEntNum, &target->DosDirNum);
  529. if (result) {
  530. return result;
  531. }
  532. target->nameLen = strlen(volname);
  533. memcpy(target->entryName, volname, target->nameLen+1);
  534. target->attributes = aDIR;
  535. /* set dates to Jan 1, 1986 00:00 */
  536. target->creationTime = target->modifyTime = cpu_to_le16(0x0000);
  537. target->creationDate = target->modifyDate = target->lastAccessDate = cpu_to_le16(0x0C21);
  538. target->nfs.mode = 0;
  539. return 0;
  540. }
  541. int ncp_modify_file_or_subdir_dos_info_path(struct ncp_server *server,
  542. struct inode *dir,
  543. const char *path,
  544. __le32 info_mask,
  545. const struct nw_modify_dos_info *info)
  546. {
  547. __u8 volnum = NCP_FINFO(dir)->volNumber;
  548. __le32 dirent = NCP_FINFO(dir)->dirEntNum;
  549. int result;
  550. ncp_init_request(server);
  551. ncp_add_byte(server, 7); /* subfunction */
  552. ncp_add_byte(server, server->name_space[volnum]);
  553. ncp_add_byte(server, 0); /* reserved */
  554. ncp_add_word(server, cpu_to_le16(0x8006)); /* search attribs: all */
  555. ncp_add_dword(server, info_mask);
  556. ncp_add_mem(server, info, sizeof(*info));
  557. ncp_add_handle_path(server, volnum, dirent, 1, path);
  558. result = ncp_request(server, 87);
  559. ncp_unlock_server(server);
  560. return result;
  561. }
  562. int ncp_modify_file_or_subdir_dos_info(struct ncp_server *server,
  563. struct inode *dir,
  564. __le32 info_mask,
  565. const struct nw_modify_dos_info *info)
  566. {
  567. return ncp_modify_file_or_subdir_dos_info_path(server, dir, NULL,
  568. info_mask, info);
  569. }
  570. #ifdef CONFIG_NCPFS_NFS_NS
  571. int ncp_modify_nfs_info(struct ncp_server *server, __u8 volnum, __le32 dirent,
  572. __u32 mode, __u32 rdev)
  573. {
  574. int result = 0;
  575. if (server->name_space[volnum] == NW_NS_NFS) {
  576. ncp_init_request(server);
  577. ncp_add_byte(server, 25); /* subfunction */
  578. ncp_add_byte(server, server->name_space[volnum]);
  579. ncp_add_byte(server, NW_NS_NFS);
  580. ncp_add_byte(server, volnum);
  581. ncp_add_dword(server, dirent);
  582. /* we must always operate on both nlinks and rdev, otherwise
  583. rdev is not set */
  584. ncp_add_dword_lh(server, NSIBM_NFS_MODE | NSIBM_NFS_NLINKS | NSIBM_NFS_RDEV);
  585. ncp_add_dword_lh(server, mode);
  586. ncp_add_dword_lh(server, 1); /* nlinks */
  587. ncp_add_dword_lh(server, rdev);
  588. result = ncp_request(server, 87);
  589. ncp_unlock_server(server);
  590. }
  591. return result;
  592. }
  593. #endif
  594. static int
  595. ncp_DeleteNSEntry(struct ncp_server *server,
  596. __u8 have_dir_base, __u8 volnum, __le32 dirent,
  597. char* name, __u8 ns, __le16 attr)
  598. {
  599. int result;
  600. ncp_init_request(server);
  601. ncp_add_byte(server, 8); /* subfunction */
  602. ncp_add_byte(server, ns);
  603. ncp_add_byte(server, 0); /* reserved */
  604. ncp_add_word(server, attr); /* search attribs: all */
  605. ncp_add_handle_path(server, volnum, dirent, have_dir_base, name);
  606. result = ncp_request(server, 87);
  607. ncp_unlock_server(server);
  608. return result;
  609. }
  610. int
  611. ncp_del_file_or_subdir2(struct ncp_server *server,
  612. struct dentry *dentry)
  613. {
  614. struct inode *inode = dentry->d_inode;
  615. __u8 volnum;
  616. __le32 dirent;
  617. if (!inode) {
  618. return 0xFF; /* Any error */
  619. }
  620. volnum = NCP_FINFO(inode)->volNumber;
  621. dirent = NCP_FINFO(inode)->DosDirNum;
  622. return ncp_DeleteNSEntry(server, 1, volnum, dirent, NULL, NW_NS_DOS, cpu_to_le16(0x8006));
  623. }
  624. int
  625. ncp_del_file_or_subdir(struct ncp_server *server,
  626. struct inode *dir, char *name)
  627. {
  628. __u8 volnum = NCP_FINFO(dir)->volNumber;
  629. __le32 dirent = NCP_FINFO(dir)->dirEntNum;
  630. #ifdef CONFIG_NCPFS_NFS_NS
  631. if (server->name_space[volnum]==NW_NS_NFS)
  632. {
  633. int result;
  634. result=ncp_obtain_DOS_dir_base(server, volnum, dirent, name, &dirent);
  635. if (result) return result;
  636. return ncp_DeleteNSEntry(server, 1, volnum, dirent, NULL, NW_NS_DOS, cpu_to_le16(0x8006));
  637. }
  638. else
  639. #endif /* CONFIG_NCPFS_NFS_NS */
  640. return ncp_DeleteNSEntry(server, 1, volnum, dirent, name, server->name_space[volnum], cpu_to_le16(0x8006));
  641. }
  642. static inline void ConvertToNWfromDWORD(__u16 v0, __u16 v1, __u8 ret[6])
  643. {
  644. __le16 *dest = (__le16 *) ret;
  645. dest[1] = cpu_to_le16(v0);
  646. dest[2] = cpu_to_le16(v1);
  647. dest[0] = cpu_to_le16(v0 + 1);
  648. return;
  649. }
  650. /* If both dir and name are NULL, then in target there's already a
  651. looked-up entry that wants to be opened. */
  652. int ncp_open_create_file_or_subdir(struct ncp_server *server,
  653. struct inode *dir, char *name,
  654. int open_create_mode,
  655. __le32 create_attributes,
  656. __le16 desired_acc_rights,
  657. struct ncp_entry_info *target)
  658. {
  659. __le16 search_attribs = cpu_to_le16(0x0006);
  660. __u8 volnum;
  661. __le32 dirent;
  662. int result;
  663. volnum = NCP_FINFO(dir)->volNumber;
  664. dirent = NCP_FINFO(dir)->dirEntNum;
  665. if ((create_attributes & aDIR) != 0) {
  666. search_attribs |= cpu_to_le16(0x8000);
  667. }
  668. ncp_init_request(server);
  669. ncp_add_byte(server, 1); /* subfunction */
  670. ncp_add_byte(server, server->name_space[volnum]);
  671. ncp_add_byte(server, open_create_mode);
  672. ncp_add_word(server, search_attribs);
  673. ncp_add_dword(server, RIM_ALL);
  674. ncp_add_dword(server, create_attributes);
  675. /* The desired acc rights seem to be the inherited rights mask
  676. for directories */
  677. ncp_add_word(server, desired_acc_rights);
  678. ncp_add_handle_path(server, volnum, dirent, 1, name);
  679. if ((result = ncp_request(server, 87)) != 0)
  680. goto out;
  681. if (!(create_attributes & aDIR))
  682. target->opened = 1;
  683. /* in target there's a new finfo to fill */
  684. ncp_extract_file_info(ncp_reply_data(server, 6), &(target->i));
  685. target->volume = target->i.volNumber;
  686. ConvertToNWfromDWORD(ncp_reply_le16(server, 0),
  687. ncp_reply_le16(server, 2),
  688. target->file_handle);
  689. ncp_unlock_server(server);
  690. (void)ncp_obtain_nfs_info(server, &(target->i));
  691. return 0;
  692. out:
  693. ncp_unlock_server(server);
  694. return result;
  695. }
  696. int
  697. ncp_initialize_search(struct ncp_server *server, struct inode *dir,
  698. struct nw_search_sequence *target)
  699. {
  700. __u8 volnum = NCP_FINFO(dir)->volNumber;
  701. __le32 dirent = NCP_FINFO(dir)->dirEntNum;
  702. int result;
  703. ncp_init_request(server);
  704. ncp_add_byte(server, 2); /* subfunction */
  705. ncp_add_byte(server, server->name_space[volnum]);
  706. ncp_add_byte(server, 0); /* reserved */
  707. ncp_add_handle_path(server, volnum, dirent, 1, NULL);
  708. result = ncp_request(server, 87);
  709. if (result)
  710. goto out;
  711. memcpy(target, ncp_reply_data(server, 0), sizeof(*target));
  712. out:
  713. ncp_unlock_server(server);
  714. return result;
  715. }
  716. int ncp_search_for_fileset(struct ncp_server *server,
  717. struct nw_search_sequence *seq,
  718. int* more,
  719. int* cnt,
  720. char* buffer,
  721. size_t bufsize,
  722. char** rbuf,
  723. size_t* rsize)
  724. {
  725. int result;
  726. ncp_init_request(server);
  727. ncp_add_byte(server, 20);
  728. ncp_add_byte(server, server->name_space[seq->volNumber]);
  729. ncp_add_byte(server, 0); /* datastream */
  730. ncp_add_word(server, cpu_to_le16(0x8006));
  731. ncp_add_dword(server, RIM_ALL);
  732. ncp_add_word(server, cpu_to_le16(32767)); /* max returned items */
  733. ncp_add_mem(server, seq, 9);
  734. #ifdef CONFIG_NCPFS_NFS_NS
  735. if (server->name_space[seq->volNumber] == NW_NS_NFS) {
  736. ncp_add_byte(server, 0); /* 0 byte pattern */
  737. } else
  738. #endif
  739. {
  740. ncp_add_byte(server, 2); /* 2 byte pattern */
  741. ncp_add_byte(server, 0xff); /* following is a wildcard */
  742. ncp_add_byte(server, '*');
  743. }
  744. result = ncp_request2(server, 87, buffer, bufsize);
  745. if (result) {
  746. ncp_unlock_server(server);
  747. return result;
  748. }
  749. if (server->ncp_reply_size < 12) {
  750. ncp_unlock_server(server);
  751. return 0xFF;
  752. }
  753. *rsize = server->ncp_reply_size - 12;
  754. ncp_unlock_server(server);
  755. buffer = buffer + sizeof(struct ncp_reply_header);
  756. *rbuf = buffer + 12;
  757. *cnt = WVAL_LH(buffer + 10);
  758. *more = BVAL(buffer + 9);
  759. memcpy(seq, buffer, 9);
  760. return 0;
  761. }
  762. static int
  763. ncp_RenameNSEntry(struct ncp_server *server,
  764. struct inode *old_dir, char *old_name, __le16 old_type,
  765. struct inode *new_dir, char *new_name)
  766. {
  767. int result = -EINVAL;
  768. if ((old_dir == NULL) || (old_name == NULL) ||
  769. (new_dir == NULL) || (new_name == NULL))
  770. goto out;
  771. ncp_init_request(server);
  772. ncp_add_byte(server, 4); /* subfunction */
  773. ncp_add_byte(server, server->name_space[NCP_FINFO(old_dir)->volNumber]);
  774. ncp_add_byte(server, 1); /* rename flag */
  775. ncp_add_word(server, old_type); /* search attributes */
  776. /* source Handle Path */
  777. ncp_add_byte(server, NCP_FINFO(old_dir)->volNumber);
  778. ncp_add_dword(server, NCP_FINFO(old_dir)->dirEntNum);
  779. ncp_add_byte(server, 1);
  780. ncp_add_byte(server, 1); /* 1 source component */
  781. /* dest Handle Path */
  782. ncp_add_byte(server, NCP_FINFO(new_dir)->volNumber);
  783. ncp_add_dword(server, NCP_FINFO(new_dir)->dirEntNum);
  784. ncp_add_byte(server, 1);
  785. ncp_add_byte(server, 1); /* 1 destination component */
  786. /* source path string */
  787. ncp_add_pstring(server, old_name);
  788. /* dest path string */
  789. ncp_add_pstring(server, new_name);
  790. result = ncp_request(server, 87);
  791. ncp_unlock_server(server);
  792. out:
  793. return result;
  794. }
  795. int ncp_ren_or_mov_file_or_subdir(struct ncp_server *server,
  796. struct inode *old_dir, char *old_name,
  797. struct inode *new_dir, char *new_name)
  798. {
  799. int result;
  800. __le16 old_type = cpu_to_le16(0x06);
  801. /* If somebody can do it atomic, call me... vandrove@vc.cvut.cz */
  802. result = ncp_RenameNSEntry(server, old_dir, old_name, old_type,
  803. new_dir, new_name);
  804. if (result == 0xFF) /* File Not Found, try directory */
  805. {
  806. old_type = cpu_to_le16(0x16);
  807. result = ncp_RenameNSEntry(server, old_dir, old_name, old_type,
  808. new_dir, new_name);
  809. }
  810. if (result != 0x92) return result; /* All except NO_FILES_RENAMED */
  811. result = ncp_del_file_or_subdir(server, new_dir, new_name);
  812. if (result != 0) return -EACCES;
  813. result = ncp_RenameNSEntry(server, old_dir, old_name, old_type,
  814. new_dir, new_name);
  815. return result;
  816. }
  817. /* We have to transfer to/from user space */
  818. int
  819. ncp_read_kernel(struct ncp_server *server, const char *file_id,
  820. __u32 offset, __u16 to_read, char *target, int *bytes_read)
  821. {
  822. char *source;
  823. int result;
  824. ncp_init_request(server);
  825. ncp_add_byte(server, 0);
  826. ncp_add_mem(server, file_id, 6);
  827. ncp_add_be32(server, offset);
  828. ncp_add_be16(server, to_read);
  829. if ((result = ncp_request(server, 72)) != 0) {
  830. goto out;
  831. }
  832. *bytes_read = ncp_reply_be16(server, 0);
  833. source = ncp_reply_data(server, 2 + (offset & 1));
  834. memcpy(target, source, *bytes_read);
  835. out:
  836. ncp_unlock_server(server);
  837. return result;
  838. }
  839. /* There is a problem... egrep and some other silly tools do:
  840. x = mmap(NULL, MAP_PRIVATE, PROT_READ|PROT_WRITE, <ncpfs fd>, 32768);
  841. read(<ncpfs fd>, x, 32768);
  842. Now copying read result by copy_to_user causes pagefault. This pagefault
  843. could not be handled because of server was locked due to read. So we have
  844. to use temporary buffer. So ncp_unlock_server must be done before
  845. copy_to_user (and for write, copy_from_user must be done before
  846. ncp_init_request... same applies for send raw packet ioctl). Because of
  847. file is normally read in bigger chunks, caller provides kmalloced
  848. (vmalloced) chunk of memory with size >= to_read...
  849. */
  850. int
  851. ncp_read_bounce(struct ncp_server *server, const char *file_id,
  852. __u32 offset, __u16 to_read, char __user *target, int *bytes_read,
  853. void* bounce, __u32 bufsize)
  854. {
  855. int result;
  856. ncp_init_request(server);
  857. ncp_add_byte(server, 0);
  858. ncp_add_mem(server, file_id, 6);
  859. ncp_add_be32(server, offset);
  860. ncp_add_be16(server, to_read);
  861. result = ncp_request2(server, 72, bounce, bufsize);
  862. ncp_unlock_server(server);
  863. if (!result) {
  864. int len = get_unaligned_be16((char *)bounce +
  865. sizeof(struct ncp_reply_header));
  866. result = -EIO;
  867. if (len <= to_read) {
  868. char* source;
  869. source = (char*)bounce +
  870. sizeof(struct ncp_reply_header) + 2 +
  871. (offset & 1);
  872. *bytes_read = len;
  873. result = 0;
  874. if (copy_to_user(target, source, len))
  875. result = -EFAULT;
  876. }
  877. }
  878. return result;
  879. }
  880. int
  881. ncp_write_kernel(struct ncp_server *server, const char *file_id,
  882. __u32 offset, __u16 to_write,
  883. const char *source, int *bytes_written)
  884. {
  885. int result;
  886. ncp_init_request(server);
  887. ncp_add_byte(server, 0);
  888. ncp_add_mem(server, file_id, 6);
  889. ncp_add_be32(server, offset);
  890. ncp_add_be16(server, to_write);
  891. ncp_add_mem(server, source, to_write);
  892. if ((result = ncp_request(server, 73)) == 0)
  893. *bytes_written = to_write;
  894. ncp_unlock_server(server);
  895. return result;
  896. }
  897. #ifdef CONFIG_NCPFS_IOCTL_LOCKING
  898. int
  899. ncp_LogPhysicalRecord(struct ncp_server *server, const char *file_id,
  900. __u8 locktype, __u32 offset, __u32 length, __u16 timeout)
  901. {
  902. int result;
  903. ncp_init_request(server);
  904. ncp_add_byte(server, locktype);
  905. ncp_add_mem(server, file_id, 6);
  906. ncp_add_be32(server, offset);
  907. ncp_add_be32(server, length);
  908. ncp_add_be16(server, timeout);
  909. if ((result = ncp_request(server, 0x1A)) != 0)
  910. {
  911. ncp_unlock_server(server);
  912. return result;
  913. }
  914. ncp_unlock_server(server);
  915. return 0;
  916. }
  917. int
  918. ncp_ClearPhysicalRecord(struct ncp_server *server, const char *file_id,
  919. __u32 offset, __u32 length)
  920. {
  921. int result;
  922. ncp_init_request(server);
  923. ncp_add_byte(server, 0); /* who knows... lanalyzer says that */
  924. ncp_add_mem(server, file_id, 6);
  925. ncp_add_be32(server, offset);
  926. ncp_add_be32(server, length);
  927. if ((result = ncp_request(server, 0x1E)) != 0)
  928. {
  929. ncp_unlock_server(server);
  930. return result;
  931. }
  932. ncp_unlock_server(server);
  933. return 0;
  934. }
  935. #endif /* CONFIG_NCPFS_IOCTL_LOCKING */
  936. #ifdef CONFIG_NCPFS_NLS
  937. /* This are the NLS conversion routines with inspirations and code parts
  938. * from the vfat file system and hints from Petr Vandrovec.
  939. */
  940. int
  941. ncp__io2vol(struct ncp_server *server, unsigned char *vname, unsigned int *vlen,
  942. const unsigned char *iname, unsigned int ilen, int cc)
  943. {
  944. struct nls_table *in = server->nls_io;
  945. struct nls_table *out = server->nls_vol;
  946. unsigned char *vname_start;
  947. unsigned char *vname_end;
  948. const unsigned char *iname_end;
  949. iname_end = iname + ilen;
  950. vname_start = vname;
  951. vname_end = vname + *vlen - 1;
  952. while (iname < iname_end) {
  953. int chl;
  954. wchar_t ec;
  955. if (NCP_IS_FLAG(server, NCP_FLAG_UTF8)) {
  956. int k;
  957. unicode_t u;
  958. k = utf8_to_utf32(iname, iname_end - iname, &u);
  959. if (k < 0 || u > MAX_WCHAR_T)
  960. return -EINVAL;
  961. iname += k;
  962. ec = u;
  963. } else {
  964. if (*iname == NCP_ESC) {
  965. int k;
  966. if (iname_end - iname < 5)
  967. goto nospec;
  968. ec = 0;
  969. for (k = 1; k < 5; k++) {
  970. unsigned char nc;
  971. nc = iname[k] - '0';
  972. if (nc >= 10) {
  973. nc -= 'A' - '0' - 10;
  974. if ((nc < 10) || (nc > 15)) {
  975. goto nospec;
  976. }
  977. }
  978. ec = (ec << 4) | nc;
  979. }
  980. iname += 5;
  981. } else {
  982. nospec:;
  983. if ( (chl = in->char2uni(iname, iname_end - iname, &ec)) < 0)
  984. return chl;
  985. iname += chl;
  986. }
  987. }
  988. /* unitoupper should be here! */
  989. chl = out->uni2char(ec, vname, vname_end - vname);
  990. if (chl < 0)
  991. return chl;
  992. /* this is wrong... */
  993. if (cc) {
  994. int chi;
  995. for (chi = 0; chi < chl; chi++){
  996. vname[chi] = ncp_toupper(out, vname[chi]);
  997. }
  998. }
  999. vname += chl;
  1000. }
  1001. *vname = 0;
  1002. *vlen = vname - vname_start;
  1003. return 0;
  1004. }
  1005. int
  1006. ncp__vol2io(struct ncp_server *server, unsigned char *iname, unsigned int *ilen,
  1007. const unsigned char *vname, unsigned int vlen, int cc)
  1008. {
  1009. struct nls_table *in = server->nls_vol;
  1010. struct nls_table *out = server->nls_io;
  1011. const unsigned char *vname_end;
  1012. unsigned char *iname_start;
  1013. unsigned char *iname_end;
  1014. unsigned char *vname_cc;
  1015. int err;
  1016. vname_cc = NULL;
  1017. if (cc) {
  1018. int i;
  1019. /* this is wrong! */
  1020. vname_cc = kmalloc(vlen, GFP_KERNEL);
  1021. if (!vname_cc)
  1022. return -ENOMEM;
  1023. for (i = 0; i < vlen; i++)
  1024. vname_cc[i] = ncp_tolower(in, vname[i]);
  1025. vname = vname_cc;
  1026. }
  1027. iname_start = iname;
  1028. iname_end = iname + *ilen - 1;
  1029. vname_end = vname + vlen;
  1030. while (vname < vname_end) {
  1031. wchar_t ec;
  1032. int chl;
  1033. if ( (chl = in->char2uni(vname, vname_end - vname, &ec)) < 0) {
  1034. err = chl;
  1035. goto quit;
  1036. }
  1037. vname += chl;
  1038. /* unitolower should be here! */
  1039. if (NCP_IS_FLAG(server, NCP_FLAG_UTF8)) {
  1040. int k;
  1041. k = utf32_to_utf8(ec, iname, iname_end - iname);
  1042. if (k < 0) {
  1043. err = -ENAMETOOLONG;
  1044. goto quit;
  1045. }
  1046. iname += k;
  1047. } else {
  1048. if ( (chl = out->uni2char(ec, iname, iname_end - iname)) >= 0) {
  1049. iname += chl;
  1050. } else {
  1051. int k;
  1052. if (iname_end - iname < 5) {
  1053. err = -ENAMETOOLONG;
  1054. goto quit;
  1055. }
  1056. *iname = NCP_ESC;
  1057. for (k = 4; k > 0; k--) {
  1058. unsigned char v;
  1059. v = (ec & 0xF) + '0';
  1060. if (v > '9') {
  1061. v += 'A' - '9' - 1;
  1062. }
  1063. iname[k] = v;
  1064. ec >>= 4;
  1065. }
  1066. iname += 5;
  1067. }
  1068. }
  1069. }
  1070. *iname = 0;
  1071. *ilen = iname - iname_start;
  1072. err = 0;
  1073. quit:;
  1074. if (cc)
  1075. kfree(vname_cc);
  1076. return err;
  1077. }
  1078. #else
  1079. int
  1080. ncp__io2vol(unsigned char *vname, unsigned int *vlen,
  1081. const unsigned char *iname, unsigned int ilen, int cc)
  1082. {
  1083. int i;
  1084. if (*vlen <= ilen)
  1085. return -ENAMETOOLONG;
  1086. if (cc)
  1087. for (i = 0; i < ilen; i++) {
  1088. *vname = toupper(*iname);
  1089. vname++;
  1090. iname++;
  1091. }
  1092. else {
  1093. memmove(vname, iname, ilen);
  1094. vname += ilen;
  1095. }
  1096. *vlen = ilen;
  1097. *vname = 0;
  1098. return 0;
  1099. }
  1100. int
  1101. ncp__vol2io(unsigned char *iname, unsigned int *ilen,
  1102. const unsigned char *vname, unsigned int vlen, int cc)
  1103. {
  1104. int i;
  1105. if (*ilen <= vlen)
  1106. return -ENAMETOOLONG;
  1107. if (cc)
  1108. for (i = 0; i < vlen; i++) {
  1109. *iname = tolower(*vname);
  1110. iname++;
  1111. vname++;
  1112. }
  1113. else {
  1114. memmove(iname, vname, vlen);
  1115. iname += vlen;
  1116. }
  1117. *ilen = vlen;
  1118. *iname = 0;
  1119. return 0;
  1120. }
  1121. #endif