|  | /* | 
|  | * (C) Copyright 2002 | 
|  | * Stäubli Faverges - <www.staubli.com> | 
|  | * Pierre AUBERT  p.aubert@staubli.com | 
|  | * | 
|  | * See file CREDITS for list of people who contributed to this | 
|  | * project. | 
|  | * | 
|  | * This program is free software; you can redistribute it and/or | 
|  | * modify it under the terms of the GNU General Public License as | 
|  | * published by the Free Software Foundation; either version 2 of | 
|  | * the License, or (at your option) any later version. | 
|  | * | 
|  | * This program is distributed in the hope that it will be useful, | 
|  | * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | * GNU General Public License for more details. | 
|  | * | 
|  | * You should have received a copy of the GNU General Public License | 
|  | * along with this program; if not, write to the Free Software | 
|  | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | 
|  | * MA 02111-1307 USA | 
|  | */ | 
|  |  | 
|  | #include <common.h> | 
|  | #include <config.h> | 
|  | #include <linux/ctype.h> | 
|  |  | 
|  | #include "dos.h" | 
|  | #include "fdos.h" | 
|  |  | 
|  | static int dir_read (Fs_t *fs, | 
|  | Slot_t *dir, | 
|  | Directory_t *dirent, | 
|  | int num, | 
|  | struct vfat_state *v); | 
|  |  | 
|  | static int unicode_read (char *in, char *out, int num); | 
|  | static int match (const char *s, const char *p); | 
|  | static unsigned char sum_shortname (char *name); | 
|  | static int check_vfat (struct vfat_state *v, Directory_t *dir); | 
|  | static char *conv_name (char *name, char *ext, char Case, char *ans); | 
|  |  | 
|  |  | 
|  | /*----------------------------------------------------------------------------- | 
|  | * clear_vfat -- | 
|  | *----------------------------------------------------------------------------- | 
|  | */ | 
|  | static void clear_vfat (struct vfat_state *v) | 
|  | { | 
|  | v -> subentries = 0; | 
|  | v -> status = 0; | 
|  | } | 
|  |  | 
|  | /*----------------------------------------------------------------------------- | 
|  | * vfat_lookup -- | 
|  | *----------------------------------------------------------------------------- | 
|  | */ | 
|  | int vfat_lookup (Slot_t *dir, | 
|  | Fs_t *fs, | 
|  | Directory_t *dirent, | 
|  | int *entry, | 
|  | int *vfat_start, | 
|  | char *filename, | 
|  | int flags, | 
|  | char *outname, | 
|  | Slot_t *file) | 
|  | { | 
|  | int found; | 
|  | struct vfat_state vfat; | 
|  | char newfile [VSE_NAMELEN]; | 
|  | int vfat_present = 0; | 
|  |  | 
|  | if (*entry == -1) { | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | found = 0; | 
|  | clear_vfat (&vfat); | 
|  | while (1) { | 
|  | if (dir_read (fs, dir, dirent, *entry, &vfat) < 0) { | 
|  | if (vfat_start) { | 
|  | *vfat_start = *entry; | 
|  | } | 
|  | break; | 
|  | } | 
|  | (*entry)++; | 
|  |  | 
|  | /* Empty slot                                                        */ | 
|  | if (dirent -> name[0] == '\0'){ | 
|  | if (vfat_start == 0) { | 
|  | break; | 
|  | } | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if (dirent -> attr == ATTR_VSE) { | 
|  | /* VSE entry, continue                                           */ | 
|  | continue; | 
|  | } | 
|  | if ( (dirent -> name [0] == DELMARK) || | 
|  | ((dirent -> attr & ATTR_DIRECTORY) != 0 && | 
|  | (flags & ACCEPT_DIR) == 0) || | 
|  | ((dirent -> attr & ATTR_VOLUME) != 0 && | 
|  | (flags & ACCEPT_LABEL) == 0) || | 
|  | (((dirent -> attr & (ATTR_DIRECTORY | ATTR_VOLUME)) == 0) && | 
|  | (flags & ACCEPT_PLAIN) == 0)) { | 
|  | clear_vfat (&vfat); | 
|  | continue; | 
|  | } | 
|  |  | 
|  | vfat_present = check_vfat (&vfat, dirent); | 
|  | if (vfat_start) { | 
|  | *vfat_start = *entry - 1; | 
|  | if (vfat_present) { | 
|  | *vfat_start -= vfat.subentries; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (dirent -> attr & ATTR_VOLUME) { | 
|  | strncpy (newfile, dirent -> name, 8); | 
|  | newfile [8] = '\0'; | 
|  | strncat (newfile, dirent -> ext, 3); | 
|  | newfile [11] = '\0'; | 
|  | } | 
|  | else { | 
|  | conv_name (dirent -> name, dirent -> ext, dirent -> Case, newfile); | 
|  | } | 
|  |  | 
|  | if (flags & MATCH_ANY) { | 
|  | found = 1; | 
|  | break; | 
|  | } | 
|  |  | 
|  | if ((vfat_present && match (vfat.name, filename)) || | 
|  | (match (newfile, filename))) { | 
|  | found = 1; | 
|  | break; | 
|  | } | 
|  | clear_vfat (&vfat); | 
|  | } | 
|  |  | 
|  | if (found) { | 
|  | if ((flags & DO_OPEN) && file) { | 
|  | if (open_file (file, dirent) < 0) { | 
|  | return (-1); | 
|  | } | 
|  | } | 
|  | if (outname) { | 
|  | if (vfat_present) { | 
|  | strcpy (outname, vfat.name); | 
|  | } | 
|  | else { | 
|  | strcpy (outname, newfile); | 
|  | } | 
|  | } | 
|  | return (0);                    /* File found                         */ | 
|  | } else { | 
|  | *entry = -1; | 
|  | return -1;                      /* File not found                    */ | 
|  | } | 
|  | } | 
|  |  | 
|  | /*----------------------------------------------------------------------------- | 
|  | * dir_read -- Read one directory entry | 
|  | *----------------------------------------------------------------------------- | 
|  | */ | 
|  | static int dir_read (Fs_t *fs, | 
|  | Slot_t *dir, | 
|  | Directory_t *dirent, | 
|  | int num, | 
|  | struct vfat_state *v) | 
|  | { | 
|  |  | 
|  | /* read the directory entry                                              */ | 
|  | if (read_file (fs, | 
|  | dir, | 
|  | (char *)dirent, | 
|  | num * MDIR_SIZE, | 
|  | MDIR_SIZE) != MDIR_SIZE) { | 
|  | return (-1); | 
|  | } | 
|  |  | 
|  | if (v && (dirent -> attr == ATTR_VSE)) { | 
|  | struct vfat_subentry *vse; | 
|  | unsigned char id, last_flag; | 
|  | char *c; | 
|  |  | 
|  | vse = (struct vfat_subentry *) dirent; | 
|  | id = vse -> id & VSE_MASK; | 
|  | last_flag = (vse -> id & VSE_LAST); | 
|  | if (id > MAX_VFAT_SUBENTRIES) { | 
|  | /* Invalid VSE entry                                             */ | 
|  | return (-1); | 
|  | } | 
|  |  | 
|  |  | 
|  | /* Decode VSE                                                        */ | 
|  | if(v -> sum != vse -> sum) { | 
|  | clear_vfat (v); | 
|  | v -> sum = vse -> sum; | 
|  | } | 
|  |  | 
|  |  | 
|  | v -> status |= 1 << (id - 1); | 
|  | if (last_flag) { | 
|  | v -> subentries = id; | 
|  | } | 
|  |  | 
|  | c = &(v -> name [VSE_NAMELEN * (id - 1)]); | 
|  | c += unicode_read (vse->text1, c, VSE1SIZE); | 
|  | c += unicode_read (vse->text2, c, VSE2SIZE); | 
|  | c += unicode_read (vse->text3, c, VSE3SIZE); | 
|  |  | 
|  | if (last_flag) { | 
|  | *c = '\0';	        /* Null terminate long name                  */ | 
|  | } | 
|  |  | 
|  | } | 
|  | return (0); | 
|  | } | 
|  |  | 
|  | /*----------------------------------------------------------------------------- | 
|  | * unicode_read -- | 
|  | *----------------------------------------------------------------------------- | 
|  | */ | 
|  | static int unicode_read (char *in, char *out, int num) | 
|  | { | 
|  | int j; | 
|  |  | 
|  | for (j = 0; j < num; ++j) { | 
|  | if (in [1]) | 
|  | *out = '_'; | 
|  | else | 
|  | *out = in [0]; | 
|  | out ++; | 
|  | in += 2; | 
|  | } | 
|  | return num; | 
|  | } | 
|  |  | 
|  | /*----------------------------------------------------------------------------- | 
|  | * match -- | 
|  | *----------------------------------------------------------------------------- | 
|  | */ | 
|  | static int match (const char *s, const char *p) | 
|  | { | 
|  |  | 
|  | for (; *p != '\0'; ) { | 
|  | if (toupper (*s) != toupper (*p)) { | 
|  | return (0); | 
|  | } | 
|  | p++; | 
|  | s++; | 
|  | } | 
|  |  | 
|  | if (*s != '\0') { | 
|  | return (0); | 
|  | } | 
|  | else { | 
|  | return (1); | 
|  | } | 
|  | } | 
|  | /*----------------------------------------------------------------------------- | 
|  | * sum_shortname -- | 
|  | *----------------------------------------------------------------------------- | 
|  | */ | 
|  | static unsigned char sum_shortname (char *name) | 
|  | { | 
|  | unsigned char sum; | 
|  | int j; | 
|  |  | 
|  | for (j = sum = 0; j < 11; ++j) { | 
|  | sum = ((sum & 1) ? 0x80 : 0) + (sum >> 1) + | 
|  | (name [j] ? name [j] : ' '); | 
|  | } | 
|  | return (sum); | 
|  | } | 
|  | /*----------------------------------------------------------------------------- | 
|  | * check_vfat -- | 
|  | * Return 1 if long name is valid, 0 else | 
|  | *----------------------------------------------------------------------------- | 
|  | */ | 
|  | static int check_vfat (struct vfat_state *v, Directory_t *dir) | 
|  | { | 
|  | char name[12]; | 
|  |  | 
|  | if (v -> subentries == 0) { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | strncpy (name, dir -> name, 8); | 
|  | strncpy (name + 8, dir -> ext, 3); | 
|  | name [11] = '\0'; | 
|  |  | 
|  | if (v -> sum != sum_shortname (name)) { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | if( (v -> status & ((1 << v -> subentries) - 1)) != | 
|  | (1 << v -> subentries) - 1) { | 
|  | return 0; | 
|  | } | 
|  | v->name [VSE_NAMELEN * v -> subentries] = 0; | 
|  |  | 
|  | return 1; | 
|  | } | 
|  | /*----------------------------------------------------------------------------- | 
|  | * conv_name -- | 
|  | *----------------------------------------------------------------------------- | 
|  | */ | 
|  | static char *conv_name (char *name, char *ext, char Case, char *ans) | 
|  | { | 
|  | char tname [9], text [4]; | 
|  | int i; | 
|  |  | 
|  | i = 0; | 
|  | while (i < 8 && name [i] != ' ' && name [i] != '\0') { | 
|  | tname [i] = name [i]; | 
|  | i++; | 
|  | } | 
|  | tname [i] = '\0'; | 
|  |  | 
|  | if (Case & BASECASE) { | 
|  | for (i = 0; i < 8 && tname [i]; i++) { | 
|  | tname [i] = tolower (tname [i]); | 
|  | } | 
|  | } | 
|  |  | 
|  | i = 0; | 
|  | while (i < 3 && ext [i] != ' ' && ext [i] != '\0') { | 
|  | text [i] = ext [i]; | 
|  | i++; | 
|  | } | 
|  | text [i] = '\0'; | 
|  |  | 
|  | if (Case & EXTCASE){ | 
|  | for (i = 0; i < 3 && text [i]; i++) { | 
|  | text [i] = tolower (text [i]); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (*text) { | 
|  | strcpy (ans, tname); | 
|  | strcat (ans, "."); | 
|  | strcat (ans, text); | 
|  | } | 
|  | else { | 
|  | strcpy(ans, tname); | 
|  | } | 
|  | return (ans); | 
|  | } |