Refactor existing HFS+ ASCII to unicode string conversion routine to split out character conversion functionality. This will be reused by the custom dentry hash and comparison routines. This approach avoids unnecessary memory allocation compared to using the string conversion routine directly in the new functions. Signed-off-by: Duane Griffin --- --- linux-2.6.21.orig/fs/hfsplus/unicode.c +++ linux-2.6.21/fs/hfsplus/unicode.c @@ -239,58 +239,97 @@ out: return res; } -int hfsplus_asc2uni(struct super_block *sb, struct hfsplus_unistr *ustr, const char *astr, int len) +/* + * Convert one or more ASCII characters into a single unicode character. + * Returns the number of ASCII characters corresponding to the unicode char. + */ +static +int asc2uc(struct super_block *sb, const char *astr, int len, wchar_t *uc) { - struct nls_table *nls = HFSPLUS_SB(sb).nls; - int size, off, decompose; - wchar_t c; - u16 outlen = 0; + int size = HFSPLUS_SB(sb).nls->char2uni(astr, len, uc); + if (size <= 0) { + *uc = '?'; + size = 1; + } + switch (*uc) { + case 0x2400: + *uc = 0; + break; + case ':': + *uc = '/'; + break; + } + return size; +} + +struct decomposed_uc { + int size; + wchar_t str[3]; +}; + +/* Decomposes a single unicode character. */ +static void decompose_uc(wchar_t uc, struct decomposed_uc *duc) +{ + int off, ii; - decompose = !(HFSPLUS_SB(sb).flags & HFSPLUS_SB_NODECOMPOSE); + duc->size = 1; + duc->str[0] = uc; + + off = hfsplus_decompose_table[(uc >> 12) & 0xf]; + if (off == 0 || off == 0xffff) + return; + + off = hfsplus_decompose_table[off + ((uc >> 8) & 0xf)]; + if (!off) + return; + + off = hfsplus_decompose_table[off + ((uc >> 4) & 0xf)]; + if (!off) + return; + + off = hfsplus_decompose_table[off + (uc & 0xf)]; + + duc->size = off & 3; + off /= 4; + for (ii = 0; ii < duc->size; ++ii) + duc->str[ii] = hfsplus_decompose_table[off++]; +} + +/* + * Convert a ASCII character(s) to a unicode character. + * The unicode character will be decomposed, if required. + * Returns the number of ASCII characters converted. + */ +static int asc2ucd(struct super_block *sb, + const char *astr, int len, struct decomposed_uc *duc) +{ + int size = asc2uc(sb, astr, len, duc->str); + bool decompose = !(HFSPLUS_SB(sb).flags & HFSPLUS_SB_NODECOMPOSE); + + if (decompose && *duc->str >= 0xc0) + decompose_uc(*duc->str, duc); + else + duc->size = 1; + + return size; +} + +int hfsplus_asc2uni(struct super_block *sb, + struct hfsplus_unistr *ustr, const char *astr, int len) +{ + u16 outlen = 0; + struct decomposed_uc duc; while (outlen < HFSPLUS_MAX_STRLEN && len > 0) { - size = nls->char2uni(astr, len, &c); - if (size <= 0) { - c = '?'; - size = 1; - } + int ii; + int size = asc2ucd(sb, astr, len, &duc); + if (outlen + duc.size > HFSPLUS_MAX_STRLEN) + break; + astr += size; len -= size; - switch (c) { - case 0x2400: - c = 0; - break; - case ':': - c = '/'; - break; - } - if (c >= 0xc0 && decompose) { - off = hfsplus_decompose_table[(c >> 12) & 0xf]; - if (!off) - goto done; - if (off == 0xffff) { - goto done; - } - off = hfsplus_decompose_table[off + ((c >> 8) & 0xf)]; - if (!off) - goto done; - off = hfsplus_decompose_table[off + ((c >> 4) & 0xf)]; - if (!off) - goto done; - off = hfsplus_decompose_table[off + (c & 0xf)]; - size = off & 3; - if (!size) - goto done; - off /= 4; - if (outlen + size > HFSPLUS_MAX_STRLEN) - break; - do { - ustr->unicode[outlen++] = cpu_to_be16(hfsplus_decompose_table[off++]); - } while (--size > 0); - continue; - } - done: - ustr->unicode[outlen++] = cpu_to_be16(c); + for (ii = 0; ii < duc.size; ++ii) + ustr->unicode[outlen++] = cpu_to_be16(duc.str[ii]); } ustr->length = cpu_to_be16(outlen); if (len > 0) -- "I never could learn to drink that blood and call it wine" - Bob Dylan - To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/