/* * Copyright (c) 2011 Oleg Romanenko * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * - Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * - Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * - The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ /** @addtogroup exfat * @{ */ /** * @file exfat_dentry.c * @brief Functions that work with exFAT directory entries. */ #include "exfat_dentry.h" #include #include #include #include #include exfat_dentry_clsf_t exfat_classify_dentry(const exfat_dentry_t *d) { switch (d->type) { case EXFAT_TYPE_VOLLABEL: return EXFAT_DENTRY_VOLLABEL; case EXFAT_TYPE_BITMAP: return EXFAT_DENTRY_BITMAP; case EXFAT_TYPE_UCTABLE: return EXFAT_DENTRY_UCTABLE; case EXFAT_TYPE_GUID: return EXFAT_DENTRY_GUID; case EXFAT_TYPE_FILE: return EXFAT_DENTRY_FILE; case EXFAT_TYPE_STREAM: return EXFAT_DENTRY_STREAM; case EXFAT_TYPE_NAME: return EXFAT_DENTRY_NAME; case EXFAT_TYPE_UNUSED: return EXFAT_DENTRY_LAST; default: if (d->type & EXFAT_TYPE_USED) return EXFAT_DENTRY_SKIP; else return EXFAT_DENTRY_FREE; } } uint16_t exfat_name_hash(const uint16_t *name, const uint16_t *uctable, size_t chars) { uint16_t hash = 0; uint16_t ch; while (*name) { if (*name < chars) ch = uint16_t_le2host(uctable[*name]); else ch = *name; name++; hash = ((hash << 15) | (hash >> 1)) + (ch & 0xff); hash = ((hash << 15) | (hash >> 1)) + (ch >> 8); } return hash; } void exfat_dentry_get_name(const exfat_name_dentry_t *name, size_t size, uint16_t *dst, size_t *offset) { size_t i = 0; while (i < EXFAT_NAME_PART_LEN && *offset < size) { dst[*offset] = uint16_t_le2host(name->name[i]); i++; (*offset)++; } dst[*offset] = '\0'; } void exfat_dentry_get_vollabel(const exfat_vollabel_dentry_t *vollabel, size_t size, uint16_t *dst) { size_t i = 0; while (i < EXFAT_VOLLABEL_LEN && i < vollabel->size && i < size) { dst[i] = uint16_t_le2host(vollabel->label[i]); i++; } dst[i] = '\0'; } bool exfat_valid_char(char32_t ch) { if (ch >= 0x01 && ch <= 0x1F) return false; switch (ch) { case '/': case '\\': case '?': case '|': case '>': case '<': case '"': case '*': case ':': return false; default: return true; } } bool exfat_valid_name(const char *name) { size_t off = 0; char32_t ch; while ((ch = str_decode(name, &off, STR_NO_LIMIT)) != 0) { if (!exfat_valid_char(ch)) return false; } return true; } /** * @} */