usr/src/grub/grub-0.97/stage2/fsys_fat.c
author Christopher Siden <chris.siden@delphix.com>
Mon, 21 May 2012 12:11:39 -0700
changeset 13700 2889e2596bd6
parent 8044 b3af80bbf173
permissions -rw-r--r--
2619 asynchronous destruction of ZFS file systems 2747 SPA versioning with zfs feature flags Reviewed by: Matt Ahrens <[email protected]> Reviewed by: George Wilson <[email protected]> Reviewed by: Richard Lowe <[email protected]> Reviewed by: Dan Kruchinin <[email protected]> Approved by: Eric Schrock <[email protected]>
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     1
/*
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     2
 *  GRUB  --  GRand Unified Bootloader
8044
b3af80bbf173 6731552 GRUB should have the ability to overlay a logo on the graphical splash screen
William Kucharski <William.Kucharski@Sun.COM>
parents: 0
diff changeset
     3
 *  Copyright (C) 2000,2001,2005   Free Software Foundation, Inc.
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     4
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     5
 *  This program is free software; you can redistribute it and/or modify
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     6
 *  it under the terms of the GNU General Public License as published by
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     7
 *  the Free Software Foundation; either version 2 of the License, or
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     8
 *  (at your option) any later version.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
     9
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    10
 *  This program is distributed in the hope that it will be useful,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    11
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    12
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    13
 *  GNU General Public License for more details.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    14
 *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    15
 *  You should have received a copy of the GNU General Public License
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    16
 *  along with this program; if not, write to the Free Software
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    17
 *  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    18
 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    19
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    20
#ifdef FSYS_FAT
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    21
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    22
#include "shared.h"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    23
#include "filesys.h"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    24
#include "fat.h"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    25
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    26
struct fat_superblock 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    27
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    28
  int fat_offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    29
  int fat_length;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    30
  int fat_size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    31
  int root_offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    32
  int root_max;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    33
  int data_offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    34
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    35
  int num_sectors;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    36
  int num_clust;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    37
  int clust_eof_marker;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    38
  int sects_per_clust;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    39
  int sectsize_bits;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    40
  int clustsize_bits;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    41
  int root_cluster;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    42
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    43
  int cached_fat;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    44
  int file_cluster;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    45
  int current_cluster_num;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    46
  int current_cluster;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    47
};
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    48
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    49
/* pointer(s) into filesystem info buffer for DOS stuff */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    50
#define FAT_SUPER ( (struct fat_superblock *) \
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    51
 		    ( FSYS_BUF + 32256) )/* 512 bytes long */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    52
#define FAT_BUF   ( FSYS_BUF + 30208 )	/* 4 sector FAT buffer */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    53
#define NAME_BUF  ( FSYS_BUF + 29184 )	/* Filename buffer (833 bytes) */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    54
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    55
#define FAT_CACHE_SIZE 2048
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    56
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    57
static __inline__ unsigned long
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    58
grub_log2 (unsigned long word)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    59
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    60
  __asm__ ("bsfl %1,%0"
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    61
	   : "=r" (word)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    62
	   : "r" (word));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    63
  return word;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    64
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    65
#define log2 grub_log2
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    66
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    67
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    68
fat_mount (void)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    69
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    70
  struct fat_bpb bpb;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    71
  __u32 magic, first_fat;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    72
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    73
  /* Check partition type for harddisk */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    74
  if (((current_drive & 0x80) || (current_slice != 0))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    75
      && ! IS_PC_SLICE_TYPE_FAT (current_slice)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    76
      && (! IS_PC_SLICE_TYPE_BSD_WITH_FS (current_slice, FS_MSDOS)))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    77
    return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    78
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    79
  /* Read bpb */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    80
  if (! devread (0, 0, sizeof (bpb), (char *) &bpb))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    81
    return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    82
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    83
  /* Check if the number of sectors per cluster is zero here, to avoid
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    84
     zero division.  */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    85
  if (bpb.sects_per_clust == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    86
    return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    87
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    88
  FAT_SUPER->sectsize_bits = log2 (FAT_CVT_U16 (bpb.bytes_per_sect));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    89
  FAT_SUPER->clustsize_bits
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    90
    = FAT_SUPER->sectsize_bits + log2 (bpb.sects_per_clust);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    91
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    92
  /* Fill in info about super block */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    93
  FAT_SUPER->num_sectors = FAT_CVT_U16 (bpb.short_sectors) 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    94
    ? FAT_CVT_U16 (bpb.short_sectors) : bpb.long_sectors;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    95
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    96
  /* FAT offset and length */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    97
  FAT_SUPER->fat_offset = FAT_CVT_U16 (bpb.reserved_sects);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    98
  FAT_SUPER->fat_length = 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
    99
    bpb.fat_length ? bpb.fat_length : bpb.fat32_length;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   100
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   101
  /* Rootdir offset and length for FAT12/16 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   102
  FAT_SUPER->root_offset = 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   103
    FAT_SUPER->fat_offset + bpb.num_fats * FAT_SUPER->fat_length;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   104
  FAT_SUPER->root_max = FAT_DIRENTRY_LENGTH * FAT_CVT_U16(bpb.dir_entries);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   105
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   106
  /* Data offset and number of clusters */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   107
  FAT_SUPER->data_offset = 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   108
    FAT_SUPER->root_offset
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   109
    + ((FAT_SUPER->root_max - 1) >> FAT_SUPER->sectsize_bits) + 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   110
  FAT_SUPER->num_clust = 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   111
    2 + ((FAT_SUPER->num_sectors - FAT_SUPER->data_offset) 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   112
	 / bpb.sects_per_clust);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   113
  FAT_SUPER->sects_per_clust = bpb.sects_per_clust;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   114
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   115
  if (!bpb.fat_length)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   116
    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   117
      /* This is a FAT32 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   118
      if (FAT_CVT_U16(bpb.dir_entries))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   119
 	return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   120
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   121
      if (bpb.flags & 0x0080)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   122
	{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   123
	  /* FAT mirroring is disabled, get active FAT */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   124
	  int active_fat = bpb.flags & 0x000f;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   125
	  if (active_fat >= bpb.num_fats)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   126
	    return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   127
	  FAT_SUPER->fat_offset += active_fat * FAT_SUPER->fat_length;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   128
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   129
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   130
      FAT_SUPER->fat_size = 8;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   131
      FAT_SUPER->root_cluster = bpb.root_cluster;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   132
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   133
      /* Yes the following is correct.  FAT32 should be called FAT28 :) */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   134
      FAT_SUPER->clust_eof_marker = 0xffffff8;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   135
    } 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   136
  else 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   137
    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   138
      if (!FAT_SUPER->root_max)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   139
 	return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   140
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   141
      FAT_SUPER->root_cluster = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   142
      if (FAT_SUPER->num_clust > FAT_MAX_12BIT_CLUST) 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   143
	{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   144
	  FAT_SUPER->fat_size = 4;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   145
	  FAT_SUPER->clust_eof_marker = 0xfff8;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   146
	} 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   147
      else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   148
	{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   149
	  FAT_SUPER->fat_size = 3;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   150
	  FAT_SUPER->clust_eof_marker = 0xff8;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   151
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   152
    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   153
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   154
  /* Now do some sanity checks */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   155
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   156
  if (FAT_CVT_U16(bpb.bytes_per_sect) != (1 << FAT_SUPER->sectsize_bits)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   157
      || FAT_CVT_U16(bpb.bytes_per_sect) != SECTOR_SIZE
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   158
      || bpb.sects_per_clust != (1 << (FAT_SUPER->clustsize_bits
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   159
 				       - FAT_SUPER->sectsize_bits))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   160
      || FAT_SUPER->num_clust <= 2
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   161
      || (FAT_SUPER->fat_size * FAT_SUPER->num_clust / (2 * SECTOR_SIZE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   162
 	  > FAT_SUPER->fat_length))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   163
    return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   164
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   165
  /* kbs: Media check on first FAT entry [ported from PUPA] */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   166
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   167
  if (!devread(FAT_SUPER->fat_offset, 0,
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   168
               sizeof(first_fat), (char *)&first_fat))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   169
    return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   170
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   171
  if (FAT_SUPER->fat_size == 8)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   172
    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   173
      first_fat &= 0x0fffffff;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   174
      magic = 0x0fffff00;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   175
    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   176
  else if (FAT_SUPER->fat_size == 4)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   177
    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   178
      first_fat &= 0x0000ffff;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   179
      magic = 0xff00;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   180
    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   181
  else
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   182
    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   183
      first_fat &= 0x00000fff;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   184
      magic = 0x0f00;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   185
    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   186
8044
b3af80bbf173 6731552 GRUB should have the ability to overlay a logo on the graphical splash screen
William Kucharski <William.Kucharski@Sun.COM>
parents: 0
diff changeset
   187
  /* Ignore the 3rd bit, because some BIOSes assigns 0xF0 to the media
b3af80bbf173 6731552 GRUB should have the ability to overlay a logo on the graphical splash screen
William Kucharski <William.Kucharski@Sun.COM>
parents: 0
diff changeset
   188
     descriptor, even if it is a so-called superfloppy (e.g. an USB key).
b3af80bbf173 6731552 GRUB should have the ability to overlay a logo on the graphical splash screen
William Kucharski <William.Kucharski@Sun.COM>
parents: 0
diff changeset
   189
     The check may be too strict for this kind of stupid BIOSes, as
b3af80bbf173 6731552 GRUB should have the ability to overlay a logo on the graphical splash screen
William Kucharski <William.Kucharski@Sun.COM>
parents: 0
diff changeset
   190
     they overwrite the media descriptor.  */
b3af80bbf173 6731552 GRUB should have the ability to overlay a logo on the graphical splash screen
William Kucharski <William.Kucharski@Sun.COM>
parents: 0
diff changeset
   191
  if ((first_fat | 0x8) != (magic | bpb.media | 0x8))
0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   192
    return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   193
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   194
  FAT_SUPER->cached_fat = - 2 * FAT_CACHE_SIZE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   195
  return 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   196
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   197
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   198
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   199
fat_read (char *buf, int len)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   200
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   201
  int logical_clust;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   202
  int offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   203
  int ret = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   204
  int size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   205
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   206
  if (FAT_SUPER->file_cluster < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   207
    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   208
      /* root directory for fat16 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   209
      size = FAT_SUPER->root_max - filepos;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   210
      if (size > len)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   211
 	size = len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   212
      if (!devread(FAT_SUPER->root_offset, filepos, size, buf))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   213
 	return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   214
      filepos += size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   215
      return size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   216
    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   217
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   218
  logical_clust = filepos >> FAT_SUPER->clustsize_bits;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   219
  offset = (filepos & ((1 << FAT_SUPER->clustsize_bits) - 1));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   220
  if (logical_clust < FAT_SUPER->current_cluster_num)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   221
    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   222
      FAT_SUPER->current_cluster_num = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   223
      FAT_SUPER->current_cluster = FAT_SUPER->file_cluster;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   224
    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   225
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   226
  while (len > 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   227
    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   228
      int sector;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   229
      while (logical_clust > FAT_SUPER->current_cluster_num)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   230
	{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   231
	  /* calculate next cluster */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   232
	  int fat_entry = 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   233
	    FAT_SUPER->current_cluster * FAT_SUPER->fat_size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   234
	  int next_cluster;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   235
	  int cached_pos = (fat_entry - FAT_SUPER->cached_fat);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   236
	  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   237
	  if (cached_pos < 0 || 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   238
	      (cached_pos + FAT_SUPER->fat_size) > 2*FAT_CACHE_SIZE)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   239
	    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   240
	      FAT_SUPER->cached_fat = (fat_entry & ~(2*SECTOR_SIZE - 1));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   241
	      cached_pos = (fat_entry - FAT_SUPER->cached_fat);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   242
	      sector = FAT_SUPER->fat_offset
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   243
		+ FAT_SUPER->cached_fat / (2*SECTOR_SIZE);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   244
	      if (!devread (sector, 0, FAT_CACHE_SIZE, (char*) FAT_BUF))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   245
		return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   246
	    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   247
	  next_cluster = * (unsigned long *) (FAT_BUF + (cached_pos >> 1));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   248
	  if (FAT_SUPER->fat_size == 3)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   249
	    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   250
	      if (cached_pos & 1)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   251
		next_cluster >>= 4;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   252
	      next_cluster &= 0xFFF;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   253
	    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   254
	  else if (FAT_SUPER->fat_size == 4)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   255
	    next_cluster &= 0xFFFF;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   256
	  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   257
	  if (next_cluster >= FAT_SUPER->clust_eof_marker)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   258
	    return ret;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   259
	  if (next_cluster < 2 || next_cluster >= FAT_SUPER->num_clust)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   260
	    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   261
	      errnum = ERR_FSYS_CORRUPT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   262
	      return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   263
	    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   264
	  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   265
	  FAT_SUPER->current_cluster = next_cluster;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   266
	  FAT_SUPER->current_cluster_num++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   267
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   268
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   269
      sector = FAT_SUPER->data_offset +
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   270
	((FAT_SUPER->current_cluster - 2) << (FAT_SUPER->clustsize_bits
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   271
 					      - FAT_SUPER->sectsize_bits));
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   272
      size = (1 << FAT_SUPER->clustsize_bits) - offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   273
      if (size > len)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   274
	size = len;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   275
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   276
      disk_read_func = disk_read_hook;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   277
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   278
      devread(sector, offset, size, buf);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   279
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   280
      disk_read_func = NULL;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   281
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   282
      len -= size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   283
      buf += size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   284
      ret += size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   285
      filepos += size;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   286
      logical_clust++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   287
      offset = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   288
    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   289
  return errnum ? 0 : ret;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   290
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   291
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   292
int
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   293
fat_dir (char *dirname)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   294
{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   295
  char *rest, ch, dir_buf[FAT_DIRENTRY_LENGTH];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   296
  char *filename = (char *) NAME_BUF;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   297
  int attrib = FAT_ATTRIB_DIR;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   298
#ifndef STAGE1_5
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   299
  int do_possibilities = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   300
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   301
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   302
  /* XXX I18N:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   303
   * the positions 2,4,6 etc are high bytes of a 16 bit unicode char 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   304
   */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   305
  static unsigned char longdir_pos[] = 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   306
  { 1, 3, 5, 7, 9, 14, 16, 18, 20, 22, 24, 28, 30 };
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   307
  int slot = -2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   308
  int alias_checksum = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   309
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   310
  FAT_SUPER->file_cluster = FAT_SUPER->root_cluster;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   311
  filepos = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   312
  FAT_SUPER->current_cluster_num = MAXINT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   313
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   314
  /* main loop to find desired directory entry */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   315
 loop:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   316
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   317
  /* if we have a real file (and we're not just printing possibilities),
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   318
     then this is where we want to exit */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   319
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   320
  if (!*dirname || isspace (*dirname))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   321
    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   322
      if (attrib & FAT_ATTRIB_DIR)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   323
	{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   324
	  errnum = ERR_BAD_FILETYPE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   325
	  return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   326
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   327
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   328
      return 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   329
    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   330
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   331
  /* continue with the file/directory name interpretation */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   332
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   333
  while (*dirname == '/')
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   334
    dirname++;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   335
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   336
  if (!(attrib & FAT_ATTRIB_DIR))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   337
    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   338
      errnum = ERR_BAD_FILETYPE;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   339
      return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   340
    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   341
  /* Directories don't have a file size */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   342
  filemax = MAXINT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   343
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   344
  for (rest = dirname; (ch = *rest) && !isspace (ch) && ch != '/'; rest++);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   345
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   346
  *rest = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   347
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   348
# ifndef STAGE1_5
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   349
  if (print_possibilities && ch != '/')
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   350
    do_possibilities = 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   351
# endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   352
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   353
  while (1)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   354
    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   355
      if (fat_read (dir_buf, FAT_DIRENTRY_LENGTH) != FAT_DIRENTRY_LENGTH
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   356
	  || dir_buf[0] == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   357
	{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   358
	  if (!errnum)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   359
	    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   360
# ifndef STAGE1_5
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   361
	      if (print_possibilities < 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   362
		{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   363
#if 0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   364
		  putchar ('\n');
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   365
#endif
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   366
		  return 1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   367
		}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   368
# endif /* STAGE1_5 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   369
	      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   370
	      errnum = ERR_FILE_NOT_FOUND;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   371
	      *rest = ch;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   372
	    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   373
	  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   374
	  return 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   375
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   376
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   377
      if (FAT_DIRENTRY_ATTRIB (dir_buf) == FAT_ATTRIB_LONGNAME)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   378
	{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   379
	  /* This is a long filename.  The filename is build from back
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   380
	   * to front and may span multiple entries.  To bind these
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   381
	   * entries together they all contain the same checksum over
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   382
	   * the short alias.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   383
	   *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   384
	   * The id field tells if this is the first entry (the last
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   385
	   * part) of the long filename, and also at which offset this
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   386
	   * belongs.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   387
	   *
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   388
	   * We just write the part of the long filename this entry
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   389
	   * describes and continue with the next dir entry.
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   390
	   */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   391
	  int i, offset;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   392
	  unsigned char id = FAT_LONGDIR_ID(dir_buf);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   393
	  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   394
	  if ((id & 0x40)) 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   395
	    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   396
	      id &= 0x3f;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   397
	      slot = id;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   398
	      filename[slot * 13] = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   399
	      alias_checksum = FAT_LONGDIR_ALIASCHECKSUM(dir_buf);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   400
	    } 
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   401
	  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   402
	  if (id != slot || slot == 0
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   403
	      || alias_checksum != FAT_LONGDIR_ALIASCHECKSUM(dir_buf))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   404
	    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   405
	      alias_checksum = -1;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   406
	      continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   407
	    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   408
	  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   409
	  slot--;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   410
	  offset = slot * 13;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   411
	  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   412
	  for (i=0; i < 13; i++)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   413
	    filename[offset+i] = dir_buf[longdir_pos[i]];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   414
	  continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   415
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   416
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   417
      if (!FAT_DIRENTRY_VALID (dir_buf))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   418
	continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   419
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   420
      if (alias_checksum != -1 && slot == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   421
	{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   422
	  int i;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   423
	  unsigned char sum;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   424
	  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   425
	  slot = -2;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   426
	  for (sum = 0, i = 0; i< 11; i++)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   427
	    sum = ((sum >> 1) | (sum << 7)) + dir_buf[i];
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   428
	  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   429
	  if (sum == alias_checksum)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   430
	    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   431
# ifndef STAGE1_5
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   432
	      if (do_possibilities)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   433
		goto print_filename;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   434
# endif /* STAGE1_5 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   435
	      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   436
	      if (substring (dirname, filename) == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   437
		break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   438
	    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   439
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   440
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   441
      /* XXX convert to 8.3 filename format here */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   442
      {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   443
	int i, j, c;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   444
	
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   445
	for (i = 0; i < 8 && (c = filename[i] = tolower (dir_buf[i]))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   446
	       && !isspace (c); i++);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   447
	
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   448
	filename[i++] = '.';
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   449
	
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   450
	for (j = 0; j < 3 && (c = filename[i + j] = tolower (dir_buf[8 + j]))
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   451
	       && !isspace (c); j++);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   452
	
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   453
	if (j == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   454
	  i--;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   455
	
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   456
	filename[i + j] = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   457
      }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   458
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   459
# ifndef STAGE1_5
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   460
      if (do_possibilities)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   461
	{
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   462
	print_filename:
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   463
	  if (substring (dirname, filename) <= 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   464
	    {
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   465
	      if (print_possibilities > 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   466
		print_possibilities = -print_possibilities;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   467
	      print_a_completion (filename);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   468
	    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   469
	  continue;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   470
	}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   471
# endif /* STAGE1_5 */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   472
      
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   473
      if (substring (dirname, filename) == 0)
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   474
	break;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   475
    }
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   476
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   477
  *(dirname = rest) = ch;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   478
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   479
  attrib = FAT_DIRENTRY_ATTRIB (dir_buf);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   480
  filemax = FAT_DIRENTRY_FILELENGTH (dir_buf);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   481
  filepos = 0;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   482
  FAT_SUPER->file_cluster = FAT_DIRENTRY_FIRST_CLUSTER (dir_buf);
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   483
  FAT_SUPER->current_cluster_num = MAXINT;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   484
  
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   485
  /* go back to main loop at top of function */
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   486
  goto loop;
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   487
}
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   488
68f95e015346 OpenSolaris Launch
stevel@tonic-gate
parents:
diff changeset
   489
#endif /* FSYS_FAT */