| /* |
| * linux/fs/hfs/trans.c |
| * |
| * Copyright (C) 1995-1997 Paul H. Hargrove |
| * This file may be distributed under the terms of the GNU General Public License. |
| * |
| * This file contains routines for converting between the Macintosh |
| * character set and various other encodings. This includes dealing |
| * with ':' vs. '/' as the path-element separator. |
| */ |
| |
| #include <linux/types.h> |
| #include <linux/nls.h> |
| |
| #include "hfs_fs.h" |
| |
| /*================ Global functions ================*/ |
| |
| /* |
| * hfs_mac2asc() |
| * |
| * Given a 'Pascal String' (a string preceded by a length byte) in |
| * the Macintosh character set produce the corresponding filename using |
| * the 'trivial' name-mangling scheme, returning the length of the |
| * mangled filename. Note that the output string is not NULL |
| * terminated. |
| * |
| * The name-mangling works as follows: |
| * The character '/', which is illegal in Linux filenames is replaced |
| * by ':' which never appears in HFS filenames. All other characters |
| * are passed unchanged from input to output. |
| */ |
| int hfs_mac2asc(struct super_block *sb, char *out, const struct hfs_name *in) |
| { |
| struct nls_table *nls_disk = HFS_SB(sb)->nls_disk; |
| struct nls_table *nls_io = HFS_SB(sb)->nls_io; |
| const char *src; |
| char *dst; |
| int srclen, dstlen, size; |
| |
| src = in->name; |
| srclen = in->len; |
| if (srclen > HFS_NAMELEN) |
| srclen = HFS_NAMELEN; |
| dst = out; |
| dstlen = HFS_MAX_NAMELEN; |
| if (nls_io) { |
| wchar_t ch; |
| |
| while (srclen > 0) { |
| if (nls_disk) { |
| size = nls_disk->char2uni(src, srclen, &ch); |
| if (size <= 0) { |
| ch = '?'; |
| size = 1; |
| } |
| src += size; |
| srclen -= size; |
| } else { |
| ch = *src++; |
| srclen--; |
| } |
| if (ch == '/') |
| ch = ':'; |
| size = nls_io->uni2char(ch, dst, dstlen); |
| if (size < 0) { |
| if (size == -ENAMETOOLONG) |
| goto out; |
| *dst = '?'; |
| size = 1; |
| } |
| dst += size; |
| dstlen -= size; |
| } |
| } else { |
| char ch; |
| |
| while (--srclen >= 0) |
| *dst++ = (ch = *src++) == '/' ? ':' : ch; |
| } |
| out: |
| return dst - out; |
| } |
| |
| /* |
| * hfs_asc2mac() |
| * |
| * Given an ASCII string (not null-terminated) and its length, |
| * generate the corresponding filename in the Macintosh character set |
| * using the 'trivial' name-mangling scheme, returning the length of |
| * the mangled filename. Note that the output string is not NULL |
| * terminated. |
| * |
| * This routine is a inverse to hfs_mac2triv(). |
| * A ':' is replaced by a '/'. |
| */ |
| void hfs_asc2mac(struct super_block *sb, struct hfs_name *out, struct qstr *in) |
| { |
| struct nls_table *nls_disk = HFS_SB(sb)->nls_disk; |
| struct nls_table *nls_io = HFS_SB(sb)->nls_io; |
| const char *src; |
| char *dst; |
| int srclen, dstlen, size; |
| |
| src = in->name; |
| srclen = in->len; |
| dst = out->name; |
| dstlen = HFS_NAMELEN; |
| if (nls_io) { |
| wchar_t ch; |
| |
| while (srclen > 0) { |
| size = nls_io->char2uni(src, srclen, &ch); |
| if (size < 0) { |
| ch = '?'; |
| size = 1; |
| } |
| src += size; |
| srclen -= size; |
| if (ch == ':') |
| ch = '/'; |
| if (nls_disk) { |
| size = nls_disk->uni2char(ch, dst, dstlen); |
| if (size < 0) { |
| if (size == -ENAMETOOLONG) |
| goto out; |
| *dst = '?'; |
| size = 1; |
| } |
| dst += size; |
| dstlen -= size; |
| } else { |
| *dst++ = ch > 0xff ? '?' : ch; |
| dstlen--; |
| } |
| } |
| } else { |
| char ch; |
| |
| if (dstlen > srclen) |
| dstlen = srclen; |
| while (--dstlen >= 0) |
| *dst++ = (ch = *src++) == ':' ? '/' : ch; |
| } |
| out: |
| out->len = dst - (char *)out->name; |
| dstlen = HFS_NAMELEN - out->len; |
| while (--dstlen >= 0) |
| *dst++ = 0; |
| } |