| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1 | /* | 
|  | 2 | *  linux/fs/isofs/joliet.c | 
|  | 3 | * | 
|  | 4 | *  (C) 1996 Gordon Chaffee | 
|  | 5 | * | 
|  | 6 | *  Joliet: Microsoft's Unicode extensions to iso9660 | 
|  | 7 | */ | 
|  | 8 |  | 
| Al Viro | 94f2f71 | 2005-04-25 18:32:12 -0700 | [diff] [blame] | 9 | #include <linux/types.h> | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 10 | #include <linux/nls.h> | 
| Al Viro | 94f2f71 | 2005-04-25 18:32:12 -0700 | [diff] [blame] | 11 | #include "isofs.h" | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 12 |  | 
|  | 13 | /* | 
| Alexey Dobriyan | 4de151d | 2006-03-22 00:13:35 +0100 | [diff] [blame] | 14 | * Convert Unicode 16 to UTF-8 or ASCII. | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 15 | */ | 
|  | 16 | static int | 
| Al Viro | d02d48d | 2005-12-24 14:33:09 -0500 | [diff] [blame] | 17 | uni16_to_x8(unsigned char *ascii, __be16 *uni, int len, struct nls_table *nls) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 18 | { | 
| Al Viro | d02d48d | 2005-12-24 14:33:09 -0500 | [diff] [blame] | 19 | __be16 *ip, ch; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 20 | unsigned char *op; | 
|  | 21 |  | 
|  | 22 | ip = uni; | 
|  | 23 | op = ascii; | 
|  | 24 |  | 
|  | 25 | while ((ch = get_unaligned(ip)) && len) { | 
|  | 26 | int llen; | 
| Al Viro | d02d48d | 2005-12-24 14:33:09 -0500 | [diff] [blame] | 27 | llen = nls->uni2char(be16_to_cpu(ch), op, NLS_MAX_CHARSET_SIZE); | 
|  | 28 | if (llen > 0) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 29 | op += llen; | 
|  | 30 | else | 
|  | 31 | *op++ = '?'; | 
|  | 32 | ip++; | 
|  | 33 |  | 
|  | 34 | len--; | 
|  | 35 | } | 
|  | 36 | *op = 0; | 
|  | 37 | return (op - ascii); | 
|  | 38 | } | 
|  | 39 |  | 
|  | 40 | /* Convert big endian wide character string to utf8 */ | 
|  | 41 | static int | 
|  | 42 | wcsntombs_be(__u8 *s, const __u8 *pwcs, int inlen, int maxlen) | 
|  | 43 | { | 
|  | 44 | const __u8 *ip; | 
|  | 45 | __u8 *op; | 
|  | 46 | int size; | 
|  | 47 | __u16 c; | 
|  | 48 |  | 
|  | 49 | op = s; | 
|  | 50 | ip = pwcs; | 
|  | 51 | while ((*ip || ip[1]) && (maxlen > 0) && (inlen > 0)) { | 
|  | 52 | c = (*ip << 8) | ip[1]; | 
|  | 53 | if (c > 0x7f) { | 
|  | 54 | size = utf8_wctomb(op, c, maxlen); | 
|  | 55 | if (size == -1) { | 
|  | 56 | /* Ignore character and move on */ | 
|  | 57 | maxlen--; | 
|  | 58 | } else { | 
|  | 59 | op += size; | 
|  | 60 | maxlen -= size; | 
|  | 61 | } | 
|  | 62 | } else { | 
|  | 63 | *op++ = (__u8) c; | 
|  | 64 | } | 
|  | 65 | ip += 2; | 
|  | 66 | inlen--; | 
|  | 67 | } | 
|  | 68 | return (op - s); | 
|  | 69 | } | 
|  | 70 |  | 
|  | 71 | int | 
|  | 72 | get_joliet_filename(struct iso_directory_record * de, unsigned char *outname, struct inode * inode) | 
|  | 73 | { | 
|  | 74 | unsigned char utf8; | 
|  | 75 | struct nls_table *nls; | 
|  | 76 | unsigned char len = 0; | 
|  | 77 |  | 
|  | 78 | utf8 = ISOFS_SB(inode->i_sb)->s_utf8; | 
|  | 79 | nls = ISOFS_SB(inode->i_sb)->s_nls_iocharset; | 
|  | 80 |  | 
|  | 81 | if (utf8) { | 
|  | 82 | len = wcsntombs_be(outname, de->name, | 
| Dave Jones | c3ed85a | 2007-07-15 23:40:03 -0700 | [diff] [blame] | 83 | de->name_len[0] >> 1, PAGE_SIZE); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 84 | } else { | 
| Al Viro | d02d48d | 2005-12-24 14:33:09 -0500 | [diff] [blame] | 85 | len = uni16_to_x8(outname, (__be16 *) de->name, | 
| Dave Jones | c3ed85a | 2007-07-15 23:40:03 -0700 | [diff] [blame] | 86 | de->name_len[0] >> 1, nls); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 87 | } | 
| Dave Jones | c3ed85a | 2007-07-15 23:40:03 -0700 | [diff] [blame] | 88 | if ((len > 2) && (outname[len-2] == ';') && (outname[len-1] == '1')) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 89 | len -= 2; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 90 |  | 
|  | 91 | /* | 
|  | 92 | * Windows doesn't like periods at the end of a name, | 
|  | 93 | * so neither do we | 
|  | 94 | */ | 
| Dave Jones | c3ed85a | 2007-07-15 23:40:03 -0700 | [diff] [blame] | 95 | while (len >= 2 && (outname[len-1] == '.')) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 96 | len--; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 97 |  | 
|  | 98 | return len; | 
|  | 99 | } |