xref: /OK3568_Linux_fs/kernel/fs/hpfs/hpfs.h (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /* SPDX-License-Identifier: GPL-2.0 */
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  *  linux/fs/hpfs/hpfs.h
4*4882a593Smuzhiyun  *
5*4882a593Smuzhiyun  *  HPFS structures by Chris Smith, 1993
6*4882a593Smuzhiyun  *
7*4882a593Smuzhiyun  *  a little bit modified by Mikulas Patocka, 1998-1999
8*4882a593Smuzhiyun  */
9*4882a593Smuzhiyun 
10*4882a593Smuzhiyun /* The paper
11*4882a593Smuzhiyun 
12*4882a593Smuzhiyun      Duncan, Roy
13*4882a593Smuzhiyun      Design goals and implementation of the new High Performance File System
14*4882a593Smuzhiyun      Microsoft Systems Journal  Sept 1989  v4 n5 p1(13)
15*4882a593Smuzhiyun 
16*4882a593Smuzhiyun    describes what HPFS looked like when it was new, and it is the source
17*4882a593Smuzhiyun    of most of the information given here.  The rest is conjecture.
18*4882a593Smuzhiyun 
19*4882a593Smuzhiyun    For definitive information on the Duncan paper, see it, not this file.
20*4882a593Smuzhiyun    For definitive information on HPFS, ask somebody else -- this is guesswork.
21*4882a593Smuzhiyun    There are certain to be many mistakes. */
22*4882a593Smuzhiyun 
23*4882a593Smuzhiyun #if !defined(__LITTLE_ENDIAN) && !defined(__BIG_ENDIAN)
24*4882a593Smuzhiyun #error unknown endian
25*4882a593Smuzhiyun #endif
26*4882a593Smuzhiyun 
27*4882a593Smuzhiyun /* Notation */
28*4882a593Smuzhiyun 
29*4882a593Smuzhiyun typedef u32 secno;			/* sector number, partition relative */
30*4882a593Smuzhiyun 
31*4882a593Smuzhiyun typedef secno dnode_secno;		/* sector number of a dnode */
32*4882a593Smuzhiyun typedef secno fnode_secno;		/* sector number of an fnode */
33*4882a593Smuzhiyun typedef secno anode_secno;		/* sector number of an anode */
34*4882a593Smuzhiyun 
35*4882a593Smuzhiyun typedef u32 time32_t;		/* 32-bit time_t type */
36*4882a593Smuzhiyun 
37*4882a593Smuzhiyun /* sector 0 */
38*4882a593Smuzhiyun 
39*4882a593Smuzhiyun /* The boot block is very like a FAT boot block, except that the
40*4882a593Smuzhiyun    29h signature byte is 28h instead, and the ID string is "HPFS". */
41*4882a593Smuzhiyun 
42*4882a593Smuzhiyun #define BB_MAGIC 0xaa55
43*4882a593Smuzhiyun 
44*4882a593Smuzhiyun struct hpfs_boot_block
45*4882a593Smuzhiyun {
46*4882a593Smuzhiyun   u8 jmp[3];
47*4882a593Smuzhiyun   u8 oem_id[8];
48*4882a593Smuzhiyun   u8 bytes_per_sector[2];	/* 512 */
49*4882a593Smuzhiyun   u8 sectors_per_cluster;
50*4882a593Smuzhiyun   u8 n_reserved_sectors[2];
51*4882a593Smuzhiyun   u8 n_fats;
52*4882a593Smuzhiyun   u8 n_rootdir_entries[2];
53*4882a593Smuzhiyun   u8 n_sectors_s[2];
54*4882a593Smuzhiyun   u8 media_byte;
55*4882a593Smuzhiyun   __le16 sectors_per_fat;
56*4882a593Smuzhiyun   __le16 sectors_per_track;
57*4882a593Smuzhiyun   __le16 heads_per_cyl;
58*4882a593Smuzhiyun   __le32 n_hidden_sectors;
59*4882a593Smuzhiyun   __le32 n_sectors_l;		/* size of partition */
60*4882a593Smuzhiyun   u8 drive_number;
61*4882a593Smuzhiyun   u8 mbz;
62*4882a593Smuzhiyun   u8 sig_28h;			/* 28h */
63*4882a593Smuzhiyun   u8 vol_serno[4];
64*4882a593Smuzhiyun   u8 vol_label[11];
65*4882a593Smuzhiyun   u8 sig_hpfs[8];		/* "HPFS    " */
66*4882a593Smuzhiyun   u8 pad[448];
67*4882a593Smuzhiyun   __le16 magic;			/* aa55 */
68*4882a593Smuzhiyun };
69*4882a593Smuzhiyun 
70*4882a593Smuzhiyun 
71*4882a593Smuzhiyun /* sector 16 */
72*4882a593Smuzhiyun 
73*4882a593Smuzhiyun /* The super block has the pointer to the root directory. */
74*4882a593Smuzhiyun 
75*4882a593Smuzhiyun #define SB_MAGIC 0xf995e849
76*4882a593Smuzhiyun 
77*4882a593Smuzhiyun struct hpfs_super_block
78*4882a593Smuzhiyun {
79*4882a593Smuzhiyun   __le32 magic;				/* f995 e849 */
80*4882a593Smuzhiyun   __le32 magic1;			/* fa53 e9c5, more magic? */
81*4882a593Smuzhiyun   u8 version;				/* version of a filesystem  usually 2 */
82*4882a593Smuzhiyun   u8 funcversion;			/* functional version - oldest version
83*4882a593Smuzhiyun   					   of filesystem that can understand
84*4882a593Smuzhiyun 					   this disk */
85*4882a593Smuzhiyun   __le16 zero;				/* 0 */
86*4882a593Smuzhiyun   __le32 root;				/* fnode of root directory */
87*4882a593Smuzhiyun   __le32 n_sectors;			/* size of filesystem */
88*4882a593Smuzhiyun   __le32 n_badblocks;			/* number of bad blocks */
89*4882a593Smuzhiyun   __le32 bitmaps;			/* pointers to free space bit maps */
90*4882a593Smuzhiyun   __le32 zero1;				/* 0 */
91*4882a593Smuzhiyun   __le32 badblocks;			/* bad block list */
92*4882a593Smuzhiyun   __le32 zero3;				/* 0 */
93*4882a593Smuzhiyun   __le32 last_chkdsk;			/* date last checked, 0 if never */
94*4882a593Smuzhiyun   __le32 last_optimize;			/* date last optimized, 0 if never */
95*4882a593Smuzhiyun   __le32 n_dir_band;			/* number of sectors in dir band */
96*4882a593Smuzhiyun   __le32 dir_band_start;			/* first sector in dir band */
97*4882a593Smuzhiyun   __le32 dir_band_end;			/* last sector in dir band */
98*4882a593Smuzhiyun   __le32 dir_band_bitmap;		/* free space map, 1 dnode per bit */
99*4882a593Smuzhiyun   u8 volume_name[32];			/* not used */
100*4882a593Smuzhiyun   __le32 user_id_table;			/* 8 preallocated sectors - user id */
101*4882a593Smuzhiyun   u32 zero6[103];			/* 0 */
102*4882a593Smuzhiyun };
103*4882a593Smuzhiyun 
104*4882a593Smuzhiyun 
105*4882a593Smuzhiyun /* sector 17 */
106*4882a593Smuzhiyun 
107*4882a593Smuzhiyun /* The spare block has pointers to spare sectors.  */
108*4882a593Smuzhiyun 
109*4882a593Smuzhiyun #define SP_MAGIC 0xf9911849
110*4882a593Smuzhiyun 
111*4882a593Smuzhiyun struct hpfs_spare_block
112*4882a593Smuzhiyun {
113*4882a593Smuzhiyun   __le32 magic;				/* f991 1849 */
114*4882a593Smuzhiyun   __le32 magic1;				/* fa52 29c5, more magic? */
115*4882a593Smuzhiyun 
116*4882a593Smuzhiyun #ifdef __LITTLE_ENDIAN
117*4882a593Smuzhiyun   u8 dirty: 1;				/* 0 clean, 1 "improperly stopped" */
118*4882a593Smuzhiyun   u8 sparedir_used: 1;			/* spare dirblks used */
119*4882a593Smuzhiyun   u8 hotfixes_used: 1;			/* hotfixes used */
120*4882a593Smuzhiyun   u8 bad_sector: 1;			/* bad sector, corrupted disk (???) */
121*4882a593Smuzhiyun   u8 bad_bitmap: 1;			/* bad bitmap */
122*4882a593Smuzhiyun   u8 fast: 1;				/* partition was fast formatted */
123*4882a593Smuzhiyun   u8 old_wrote: 1;			/* old version wrote to partition */
124*4882a593Smuzhiyun   u8 old_wrote_1: 1;			/* old version wrote to partition (?) */
125*4882a593Smuzhiyun #else
126*4882a593Smuzhiyun   u8 old_wrote_1: 1;			/* old version wrote to partition (?) */
127*4882a593Smuzhiyun   u8 old_wrote: 1;			/* old version wrote to partition */
128*4882a593Smuzhiyun   u8 fast: 1;				/* partition was fast formatted */
129*4882a593Smuzhiyun   u8 bad_bitmap: 1;			/* bad bitmap */
130*4882a593Smuzhiyun   u8 bad_sector: 1;			/* bad sector, corrupted disk (???) */
131*4882a593Smuzhiyun   u8 hotfixes_used: 1;			/* hotfixes used */
132*4882a593Smuzhiyun   u8 sparedir_used: 1;			/* spare dirblks used */
133*4882a593Smuzhiyun   u8 dirty: 1;				/* 0 clean, 1 "improperly stopped" */
134*4882a593Smuzhiyun #endif
135*4882a593Smuzhiyun 
136*4882a593Smuzhiyun #ifdef __LITTLE_ENDIAN
137*4882a593Smuzhiyun   u8 install_dasd_limits: 1;		/* HPFS386 flags */
138*4882a593Smuzhiyun   u8 resynch_dasd_limits: 1;
139*4882a593Smuzhiyun   u8 dasd_limits_operational: 1;
140*4882a593Smuzhiyun   u8 multimedia_active: 1;
141*4882a593Smuzhiyun   u8 dce_acls_active: 1;
142*4882a593Smuzhiyun   u8 dasd_limits_dirty: 1;
143*4882a593Smuzhiyun   u8 flag67: 2;
144*4882a593Smuzhiyun #else
145*4882a593Smuzhiyun   u8 flag67: 2;
146*4882a593Smuzhiyun   u8 dasd_limits_dirty: 1;
147*4882a593Smuzhiyun   u8 dce_acls_active: 1;
148*4882a593Smuzhiyun   u8 multimedia_active: 1;
149*4882a593Smuzhiyun   u8 dasd_limits_operational: 1;
150*4882a593Smuzhiyun   u8 resynch_dasd_limits: 1;
151*4882a593Smuzhiyun   u8 install_dasd_limits: 1;		/* HPFS386 flags */
152*4882a593Smuzhiyun #endif
153*4882a593Smuzhiyun 
154*4882a593Smuzhiyun   u8 mm_contlgulty;
155*4882a593Smuzhiyun   u8 unused;
156*4882a593Smuzhiyun 
157*4882a593Smuzhiyun   __le32 hotfix_map;			/* info about remapped bad sectors */
158*4882a593Smuzhiyun   __le32 n_spares_used;			/* number of hotfixes */
159*4882a593Smuzhiyun   __le32 n_spares;			/* number of spares in hotfix map */
160*4882a593Smuzhiyun   __le32 n_dnode_spares_free;		/* spare dnodes unused */
161*4882a593Smuzhiyun   __le32 n_dnode_spares;		/* length of spare_dnodes[] list,
162*4882a593Smuzhiyun 					   follows in this block*/
163*4882a593Smuzhiyun   __le32 code_page_dir;			/* code page directory block */
164*4882a593Smuzhiyun   __le32 n_code_pages;			/* number of code pages */
165*4882a593Smuzhiyun   __le32 super_crc;			/* on HPFS386 and LAN Server this is
166*4882a593Smuzhiyun   					   checksum of superblock, on normal
167*4882a593Smuzhiyun 					   OS/2 unused */
168*4882a593Smuzhiyun   __le32 spare_crc;			/* on HPFS386 checksum of spareblock */
169*4882a593Smuzhiyun   __le32 zero1[15];			/* unused */
170*4882a593Smuzhiyun   __le32 spare_dnodes[100];		/* emergency free dnode list */
171*4882a593Smuzhiyun   __le32 zero2[1];			/* room for more? */
172*4882a593Smuzhiyun };
173*4882a593Smuzhiyun 
174*4882a593Smuzhiyun /* The bad block list is 4 sectors long.  The first word must be zero,
175*4882a593Smuzhiyun    the remaining words give n_badblocks bad block numbers.
176*4882a593Smuzhiyun    I bet you can see it coming... */
177*4882a593Smuzhiyun 
178*4882a593Smuzhiyun #define BAD_MAGIC 0
179*4882a593Smuzhiyun 
180*4882a593Smuzhiyun /* The hotfix map is 4 sectors long.  It looks like
181*4882a593Smuzhiyun 
182*4882a593Smuzhiyun        secno from[n_spares];
183*4882a593Smuzhiyun        secno to[n_spares];
184*4882a593Smuzhiyun 
185*4882a593Smuzhiyun    The to[] list is initialized to point to n_spares preallocated empty
186*4882a593Smuzhiyun    sectors.  The from[] list contains the sector numbers of bad blocks
187*4882a593Smuzhiyun    which have been remapped to corresponding sectors in the to[] list.
188*4882a593Smuzhiyun    n_spares_used gives the length of the from[] list. */
189*4882a593Smuzhiyun 
190*4882a593Smuzhiyun 
191*4882a593Smuzhiyun /* Sectors 18 and 19 are preallocated and unused.
192*4882a593Smuzhiyun    Maybe they're spares for 16 and 17, but simple substitution fails. */
193*4882a593Smuzhiyun 
194*4882a593Smuzhiyun 
195*4882a593Smuzhiyun /* The code page info pointed to by the spare block consists of an index
196*4882a593Smuzhiyun    block and blocks containing uppercasing tables.  I don't know what
197*4882a593Smuzhiyun    these are for (CHKDSK, maybe?) -- OS/2 does not seem to use them
198*4882a593Smuzhiyun    itself.  Linux doesn't use them either. */
199*4882a593Smuzhiyun 
200*4882a593Smuzhiyun /* block pointed to by spareblock->code_page_dir */
201*4882a593Smuzhiyun 
202*4882a593Smuzhiyun #define CP_DIR_MAGIC 0x494521f7
203*4882a593Smuzhiyun 
204*4882a593Smuzhiyun struct code_page_directory
205*4882a593Smuzhiyun {
206*4882a593Smuzhiyun   __le32 magic;				/* 4945 21f7 */
207*4882a593Smuzhiyun   __le32 n_code_pages;			/* number of pointers following */
208*4882a593Smuzhiyun   __le32 zero1[2];
209*4882a593Smuzhiyun   struct {
210*4882a593Smuzhiyun     __le16 ix;				/* index */
211*4882a593Smuzhiyun     __le16 code_page_number;		/* code page number */
212*4882a593Smuzhiyun     __le32 bounds;			/* matches corresponding word
213*4882a593Smuzhiyun 					   in data block */
214*4882a593Smuzhiyun     __le32 code_page_data;		/* sector number of a code_page_data
215*4882a593Smuzhiyun 					   containing c.p. array */
216*4882a593Smuzhiyun     __le16 index;			/* index in c.p. array in that sector*/
217*4882a593Smuzhiyun     __le16 unknown;			/* some unknown value; usually 0;
218*4882a593Smuzhiyun     					   2 in Japanese version */
219*4882a593Smuzhiyun   } array[31];				/* unknown length */
220*4882a593Smuzhiyun };
221*4882a593Smuzhiyun 
222*4882a593Smuzhiyun /* blocks pointed to by code_page_directory */
223*4882a593Smuzhiyun 
224*4882a593Smuzhiyun #define CP_DATA_MAGIC 0x894521f7
225*4882a593Smuzhiyun 
226*4882a593Smuzhiyun struct code_page_data
227*4882a593Smuzhiyun {
228*4882a593Smuzhiyun   __le32 magic;				/* 8945 21f7 */
229*4882a593Smuzhiyun   __le32 n_used;			/* # elements used in c_p_data[] */
230*4882a593Smuzhiyun   __le32 bounds[3];			/* looks a bit like
231*4882a593Smuzhiyun 					     (beg1,end1), (beg2,end2)
232*4882a593Smuzhiyun 					   one byte each */
233*4882a593Smuzhiyun   __le16 offs[3];			/* offsets from start of sector
234*4882a593Smuzhiyun 					   to start of c_p_data[ix] */
235*4882a593Smuzhiyun   struct {
236*4882a593Smuzhiyun     __le16 ix;				/* index */
237*4882a593Smuzhiyun     __le16 code_page_number;		/* code page number */
238*4882a593Smuzhiyun     __le16 unknown;			/* the same as in cp directory */
239*4882a593Smuzhiyun     u8 map[128];			/* upcase table for chars 80..ff */
240*4882a593Smuzhiyun     __le16 zero2;
241*4882a593Smuzhiyun   } code_page[3];
242*4882a593Smuzhiyun   u8 incognita[78];
243*4882a593Smuzhiyun };
244*4882a593Smuzhiyun 
245*4882a593Smuzhiyun 
246*4882a593Smuzhiyun /* Free space bitmaps are 4 sectors long, which is 16384 bits.
247*4882a593Smuzhiyun    16384 sectors is 8 meg, and each 8 meg band has a 4-sector bitmap.
248*4882a593Smuzhiyun    Bit order in the maps is little-endian.  0 means taken, 1 means free.
249*4882a593Smuzhiyun 
250*4882a593Smuzhiyun    Bit map sectors are marked allocated in the bit maps, and so are sectors
251*4882a593Smuzhiyun    off the end of the partition.
252*4882a593Smuzhiyun 
253*4882a593Smuzhiyun    Band 0 is sectors 0-3fff, its map is in sectors 18-1b.
254*4882a593Smuzhiyun    Band 1 is 4000-7fff, its map is in 7ffc-7fff.
255*4882a593Smuzhiyun    Band 2 is 8000-ffff, its map is in 8000-8003.
256*4882a593Smuzhiyun    The remaining bands have maps in their first (even) or last (odd) 4 sectors
257*4882a593Smuzhiyun      -- if the last, partial, band is odd its map is in its last 4 sectors.
258*4882a593Smuzhiyun 
259*4882a593Smuzhiyun    The bitmap locations are given in a table pointed to by the super block.
260*4882a593Smuzhiyun    No doubt they aren't constrained to be at 18, 7ffc, 8000, ...; that is
261*4882a593Smuzhiyun    just where they usually are.
262*4882a593Smuzhiyun 
263*4882a593Smuzhiyun    The "directory band" is a bunch of sectors preallocated for dnodes.
264*4882a593Smuzhiyun    It has a 4-sector free space bitmap of its own.  Each bit in the map
265*4882a593Smuzhiyun    corresponds to one 4-sector dnode, bit 0 of the map corresponding to
266*4882a593Smuzhiyun    the first 4 sectors of the directory band.  The entire band is marked
267*4882a593Smuzhiyun    allocated in the main bitmap.   The super block gives the locations
268*4882a593Smuzhiyun    of the directory band and its bitmap.  ("band" doesn't mean it is
269*4882a593Smuzhiyun    8 meg long; it isn't.)  */
270*4882a593Smuzhiyun 
271*4882a593Smuzhiyun 
272*4882a593Smuzhiyun /* dnode: directory.  4 sectors long */
273*4882a593Smuzhiyun 
274*4882a593Smuzhiyun /* A directory is a tree of dnodes.  The fnode for a directory
275*4882a593Smuzhiyun    contains one pointer, to the root dnode of the tree.  The fnode
276*4882a593Smuzhiyun    never moves, the dnodes do the B-tree thing, splitting and merging
277*4882a593Smuzhiyun    as files are added and removed.  */
278*4882a593Smuzhiyun 
279*4882a593Smuzhiyun #define DNODE_MAGIC   0x77e40aae
280*4882a593Smuzhiyun 
281*4882a593Smuzhiyun struct dnode {
282*4882a593Smuzhiyun   __le32 magic;				/* 77e4 0aae */
283*4882a593Smuzhiyun   __le32 first_free;			/* offset from start of dnode to
284*4882a593Smuzhiyun 					   first free dir entry */
285*4882a593Smuzhiyun #ifdef __LITTLE_ENDIAN
286*4882a593Smuzhiyun   u8 root_dnode: 1;			/* Is it root dnode? */
287*4882a593Smuzhiyun   u8 increment_me: 7;			/* some kind of activity counter? */
288*4882a593Smuzhiyun 					/* Neither HPFS.IFS nor CHKDSK cares
289*4882a593Smuzhiyun 					   if you change this word */
290*4882a593Smuzhiyun #else
291*4882a593Smuzhiyun   u8 increment_me: 7;			/* some kind of activity counter? */
292*4882a593Smuzhiyun 					/* Neither HPFS.IFS nor CHKDSK cares
293*4882a593Smuzhiyun 					   if you change this word */
294*4882a593Smuzhiyun   u8 root_dnode: 1;			/* Is it root dnode? */
295*4882a593Smuzhiyun #endif
296*4882a593Smuzhiyun   u8 increment_me2[3];
297*4882a593Smuzhiyun   __le32 up;				/* (root dnode) directory's fnode
298*4882a593Smuzhiyun 					   (nonroot) parent dnode */
299*4882a593Smuzhiyun   __le32 self;			/* pointer to this dnode */
300*4882a593Smuzhiyun   u8 dirent[2028];			/* one or more dirents */
301*4882a593Smuzhiyun };
302*4882a593Smuzhiyun 
303*4882a593Smuzhiyun struct hpfs_dirent {
304*4882a593Smuzhiyun   __le16 length;			/* offset to next dirent */
305*4882a593Smuzhiyun 
306*4882a593Smuzhiyun #ifdef __LITTLE_ENDIAN
307*4882a593Smuzhiyun   u8 first: 1;				/* set on phony ^A^A (".") entry */
308*4882a593Smuzhiyun   u8 has_acl: 1;
309*4882a593Smuzhiyun   u8 down: 1;				/* down pointer present (after name) */
310*4882a593Smuzhiyun   u8 last: 1;				/* set on phony \377 entry */
311*4882a593Smuzhiyun   u8 has_ea: 1;				/* entry has EA */
312*4882a593Smuzhiyun   u8 has_xtd_perm: 1;			/* has extended perm list (???) */
313*4882a593Smuzhiyun   u8 has_explicit_acl: 1;
314*4882a593Smuzhiyun   u8 has_needea: 1;			/* ?? some EA has NEEDEA set
315*4882a593Smuzhiyun 					   I have no idea why this is
316*4882a593Smuzhiyun 					   interesting in a dir entry */
317*4882a593Smuzhiyun #else
318*4882a593Smuzhiyun   u8 has_needea: 1;			/* ?? some EA has NEEDEA set
319*4882a593Smuzhiyun 					   I have no idea why this is
320*4882a593Smuzhiyun 					   interesting in a dir entry */
321*4882a593Smuzhiyun   u8 has_explicit_acl: 1;
322*4882a593Smuzhiyun   u8 has_xtd_perm: 1;			/* has extended perm list (???) */
323*4882a593Smuzhiyun   u8 has_ea: 1;				/* entry has EA */
324*4882a593Smuzhiyun   u8 last: 1;				/* set on phony \377 entry */
325*4882a593Smuzhiyun   u8 down: 1;				/* down pointer present (after name) */
326*4882a593Smuzhiyun   u8 has_acl: 1;
327*4882a593Smuzhiyun   u8 first: 1;				/* set on phony ^A^A (".") entry */
328*4882a593Smuzhiyun #endif
329*4882a593Smuzhiyun 
330*4882a593Smuzhiyun #ifdef __LITTLE_ENDIAN
331*4882a593Smuzhiyun   u8 read_only: 1;			/* dos attrib */
332*4882a593Smuzhiyun   u8 hidden: 1;				/* dos attrib */
333*4882a593Smuzhiyun   u8 system: 1;				/* dos attrib */
334*4882a593Smuzhiyun   u8 flag11: 1;				/* would be volume label dos attrib */
335*4882a593Smuzhiyun   u8 directory: 1;			/* dos attrib */
336*4882a593Smuzhiyun   u8 archive: 1;			/* dos attrib */
337*4882a593Smuzhiyun   u8 not_8x3: 1;			/* name is not 8.3 */
338*4882a593Smuzhiyun   u8 flag15: 1;
339*4882a593Smuzhiyun #else
340*4882a593Smuzhiyun   u8 flag15: 1;
341*4882a593Smuzhiyun   u8 not_8x3: 1;			/* name is not 8.3 */
342*4882a593Smuzhiyun   u8 archive: 1;			/* dos attrib */
343*4882a593Smuzhiyun   u8 directory: 1;			/* dos attrib */
344*4882a593Smuzhiyun   u8 flag11: 1;				/* would be volume label dos attrib */
345*4882a593Smuzhiyun   u8 system: 1;				/* dos attrib */
346*4882a593Smuzhiyun   u8 hidden: 1;				/* dos attrib */
347*4882a593Smuzhiyun   u8 read_only: 1;			/* dos attrib */
348*4882a593Smuzhiyun #endif
349*4882a593Smuzhiyun 
350*4882a593Smuzhiyun   __le32 fnode;				/* fnode giving allocation info */
351*4882a593Smuzhiyun   __le32 write_date;			/* mtime */
352*4882a593Smuzhiyun   __le32 file_size;			/* file length, bytes */
353*4882a593Smuzhiyun   __le32 read_date;			/* atime */
354*4882a593Smuzhiyun   __le32 creation_date;			/* ctime */
355*4882a593Smuzhiyun   __le32 ea_size;			/* total EA length, bytes */
356*4882a593Smuzhiyun   u8 no_of_acls;			/* number of ACL's (low 3 bits) */
357*4882a593Smuzhiyun   u8 ix;				/* code page index (of filename), see
358*4882a593Smuzhiyun 					   struct code_page_data */
359*4882a593Smuzhiyun   u8 namelen, name[1];			/* file name */
360*4882a593Smuzhiyun   /* dnode_secno down;	  btree down pointer, if present,
361*4882a593Smuzhiyun      			  follows name on next word boundary, or maybe it
362*4882a593Smuzhiyun 			  precedes next dirent, which is on a word boundary. */
363*4882a593Smuzhiyun };
364*4882a593Smuzhiyun 
365*4882a593Smuzhiyun 
366*4882a593Smuzhiyun /* B+ tree: allocation info in fnodes and anodes */
367*4882a593Smuzhiyun 
368*4882a593Smuzhiyun /* dnodes point to fnodes which are responsible for listing the sectors
369*4882a593Smuzhiyun    assigned to the file.  This is done with trees of (length,address)
370*4882a593Smuzhiyun    pairs.  (Actually triples, of (length, file-address, disk-address)
371*4882a593Smuzhiyun    which can represent holes.  Find out if HPFS does that.)
372*4882a593Smuzhiyun    At any rate, fnodes contain a small tree; if subtrees are needed
373*4882a593Smuzhiyun    they occupy essentially a full block in anodes.  A leaf-level tree node
374*4882a593Smuzhiyun    has 3-word entries giving sector runs, a non-leaf node has 2-word
375*4882a593Smuzhiyun    entries giving subtree pointers.  A flag in the header says which. */
376*4882a593Smuzhiyun 
377*4882a593Smuzhiyun struct bplus_leaf_node
378*4882a593Smuzhiyun {
379*4882a593Smuzhiyun   __le32 file_secno;			/* first file sector in extent */
380*4882a593Smuzhiyun   __le32 length;			/* length, sectors */
381*4882a593Smuzhiyun   __le32 disk_secno;			/* first corresponding disk sector */
382*4882a593Smuzhiyun };
383*4882a593Smuzhiyun 
384*4882a593Smuzhiyun struct bplus_internal_node
385*4882a593Smuzhiyun {
386*4882a593Smuzhiyun   __le32 file_secno;			/* subtree maps sectors < this  */
387*4882a593Smuzhiyun   __le32 down;				/* pointer to subtree */
388*4882a593Smuzhiyun };
389*4882a593Smuzhiyun 
390*4882a593Smuzhiyun enum {
391*4882a593Smuzhiyun 	BP_hbff = 1,
392*4882a593Smuzhiyun 	BP_fnode_parent = 0x20,
393*4882a593Smuzhiyun 	BP_binary_search = 0x40,
394*4882a593Smuzhiyun 	BP_internal = 0x80
395*4882a593Smuzhiyun };
396*4882a593Smuzhiyun struct bplus_header
397*4882a593Smuzhiyun {
398*4882a593Smuzhiyun   u8 flags;				/* bit 0 - high bit of first free entry offset
399*4882a593Smuzhiyun 					   bit 5 - we're pointed to by an fnode,
400*4882a593Smuzhiyun 					   the data btree or some ea or the
401*4882a593Smuzhiyun 					   main ea bootage pointer ea_secno
402*4882a593Smuzhiyun 					   bit 6 - suggest binary search (unused)
403*4882a593Smuzhiyun 					   bit 7 - 1 -> (internal) tree of anodes
404*4882a593Smuzhiyun 						   0 -> (leaf) list of extents */
405*4882a593Smuzhiyun   u8 fill[3];
406*4882a593Smuzhiyun   u8 n_free_nodes;			/* free nodes in following array */
407*4882a593Smuzhiyun   u8 n_used_nodes;			/* used nodes in following array */
408*4882a593Smuzhiyun   __le16 first_free;			/* offset from start of header to
409*4882a593Smuzhiyun 					   first free node in array */
410*4882a593Smuzhiyun   union {
411*4882a593Smuzhiyun     struct bplus_internal_node internal[0]; /* (internal) 2-word entries giving
412*4882a593Smuzhiyun 					       subtree pointers */
413*4882a593Smuzhiyun     struct bplus_leaf_node external[0];	    /* (external) 3-word entries giving
414*4882a593Smuzhiyun 					       sector runs */
415*4882a593Smuzhiyun   } u;
416*4882a593Smuzhiyun };
417*4882a593Smuzhiyun 
bp_internal(struct bplus_header * bp)418*4882a593Smuzhiyun static inline bool bp_internal(struct bplus_header *bp)
419*4882a593Smuzhiyun {
420*4882a593Smuzhiyun 	return bp->flags & BP_internal;
421*4882a593Smuzhiyun }
422*4882a593Smuzhiyun 
bp_fnode_parent(struct bplus_header * bp)423*4882a593Smuzhiyun static inline bool bp_fnode_parent(struct bplus_header *bp)
424*4882a593Smuzhiyun {
425*4882a593Smuzhiyun 	return bp->flags & BP_fnode_parent;
426*4882a593Smuzhiyun }
427*4882a593Smuzhiyun 
428*4882a593Smuzhiyun /* fnode: root of allocation b+ tree, and EA's */
429*4882a593Smuzhiyun 
430*4882a593Smuzhiyun /* Every file and every directory has one fnode, pointed to by the directory
431*4882a593Smuzhiyun    entry and pointing to the file's sectors or directory's root dnode.  EA's
432*4882a593Smuzhiyun    are also stored here, and there are said to be ACL's somewhere here too. */
433*4882a593Smuzhiyun 
434*4882a593Smuzhiyun #define FNODE_MAGIC 0xf7e40aae
435*4882a593Smuzhiyun 
436*4882a593Smuzhiyun enum {FNODE_anode = cpu_to_le16(2), FNODE_dir = cpu_to_le16(256)};
437*4882a593Smuzhiyun struct fnode
438*4882a593Smuzhiyun {
439*4882a593Smuzhiyun   __le32 magic;				/* f7e4 0aae */
440*4882a593Smuzhiyun   __le32 zero1[2];			/* read history */
441*4882a593Smuzhiyun   u8 len, name[15];			/* true length, truncated name */
442*4882a593Smuzhiyun   __le32 up;				/* pointer to file's directory fnode */
443*4882a593Smuzhiyun   __le32 acl_size_l;
444*4882a593Smuzhiyun   __le32 acl_secno;
445*4882a593Smuzhiyun   __le16 acl_size_s;
446*4882a593Smuzhiyun   u8 acl_anode;
447*4882a593Smuzhiyun   u8 zero2;				/* history bit count */
448*4882a593Smuzhiyun   __le32 ea_size_l;			/* length of disk-resident ea's */
449*4882a593Smuzhiyun   __le32 ea_secno;			/* first sector of disk-resident ea's*/
450*4882a593Smuzhiyun   __le16 ea_size_s;			/* length of fnode-resident ea's */
451*4882a593Smuzhiyun 
452*4882a593Smuzhiyun   __le16 flags;				/* bit 1 set -> ea_secno is an anode */
453*4882a593Smuzhiyun 					/* bit 8 set -> directory.  first & only extent
454*4882a593Smuzhiyun 					   points to dnode. */
455*4882a593Smuzhiyun   struct bplus_header btree;		/* b+ tree, 8 extents or 12 subtrees */
456*4882a593Smuzhiyun   union {
457*4882a593Smuzhiyun     struct bplus_leaf_node external[8];
458*4882a593Smuzhiyun     struct bplus_internal_node internal[12];
459*4882a593Smuzhiyun   } u;
460*4882a593Smuzhiyun 
461*4882a593Smuzhiyun   __le32 file_size;			/* file length, bytes */
462*4882a593Smuzhiyun   __le32 n_needea;			/* number of EA's with NEEDEA set */
463*4882a593Smuzhiyun   u8 user_id[16];			/* unused */
464*4882a593Smuzhiyun   __le16 ea_offs;			/* offset from start of fnode
465*4882a593Smuzhiyun 					   to first fnode-resident ea */
466*4882a593Smuzhiyun   u8 dasd_limit_treshhold;
467*4882a593Smuzhiyun   u8 dasd_limit_delta;
468*4882a593Smuzhiyun   __le32 dasd_limit;
469*4882a593Smuzhiyun   __le32 dasd_usage;
470*4882a593Smuzhiyun   u8 ea[316];				/* zero or more EA's, packed together
471*4882a593Smuzhiyun 					   with no alignment padding.
472*4882a593Smuzhiyun 					   (Do not use this name, get here
473*4882a593Smuzhiyun 					   via fnode + ea_offs. I think.) */
474*4882a593Smuzhiyun };
475*4882a593Smuzhiyun 
fnode_in_anode(struct fnode * p)476*4882a593Smuzhiyun static inline bool fnode_in_anode(struct fnode *p)
477*4882a593Smuzhiyun {
478*4882a593Smuzhiyun 	return (p->flags & FNODE_anode) != 0;
479*4882a593Smuzhiyun }
480*4882a593Smuzhiyun 
fnode_is_dir(struct fnode * p)481*4882a593Smuzhiyun static inline bool fnode_is_dir(struct fnode *p)
482*4882a593Smuzhiyun {
483*4882a593Smuzhiyun 	return (p->flags & FNODE_dir) != 0;
484*4882a593Smuzhiyun }
485*4882a593Smuzhiyun 
486*4882a593Smuzhiyun 
487*4882a593Smuzhiyun /* anode: 99.44% pure allocation tree */
488*4882a593Smuzhiyun 
489*4882a593Smuzhiyun #define ANODE_MAGIC 0x37e40aae
490*4882a593Smuzhiyun 
491*4882a593Smuzhiyun struct anode
492*4882a593Smuzhiyun {
493*4882a593Smuzhiyun   __le32 magic;				/* 37e4 0aae */
494*4882a593Smuzhiyun   __le32 self;				/* pointer to this anode */
495*4882a593Smuzhiyun   __le32 up;				/* parent anode or fnode */
496*4882a593Smuzhiyun 
497*4882a593Smuzhiyun   struct bplus_header btree;		/* b+tree, 40 extents or 60 subtrees */
498*4882a593Smuzhiyun   union {
499*4882a593Smuzhiyun     struct bplus_leaf_node external[40];
500*4882a593Smuzhiyun     struct bplus_internal_node internal[60];
501*4882a593Smuzhiyun   } u;
502*4882a593Smuzhiyun 
503*4882a593Smuzhiyun   __le32 fill[3];			/* unused */
504*4882a593Smuzhiyun };
505*4882a593Smuzhiyun 
506*4882a593Smuzhiyun 
507*4882a593Smuzhiyun /* extended attributes.
508*4882a593Smuzhiyun 
509*4882a593Smuzhiyun    A file's EA info is stored as a list of (name,value) pairs.  It is
510*4882a593Smuzhiyun    usually in the fnode, but (if it's large) it is moved to a single
511*4882a593Smuzhiyun    sector run outside the fnode, or to multiple runs with an anode tree
512*4882a593Smuzhiyun    that points to them.
513*4882a593Smuzhiyun 
514*4882a593Smuzhiyun    The value of a single EA is stored along with the name, or (if large)
515*4882a593Smuzhiyun    it is moved to a single sector run, or multiple runs pointed to by an
516*4882a593Smuzhiyun    anode tree, pointed to by the value field of the (name,value) pair.
517*4882a593Smuzhiyun 
518*4882a593Smuzhiyun    Flags in the EA tell whether the value is immediate, in a single sector
519*4882a593Smuzhiyun    run, or in multiple runs.  Flags in the fnode tell whether the EA list
520*4882a593Smuzhiyun    is immediate, in a single run, or in multiple runs. */
521*4882a593Smuzhiyun 
522*4882a593Smuzhiyun enum {EA_indirect = 1, EA_anode = 2, EA_needea = 128 };
523*4882a593Smuzhiyun struct extended_attribute
524*4882a593Smuzhiyun {
525*4882a593Smuzhiyun   u8 flags;				/* bit 0 set -> value gives sector number
526*4882a593Smuzhiyun 					   where real value starts */
527*4882a593Smuzhiyun 					/* bit 1 set -> sector is an anode
528*4882a593Smuzhiyun 					   that points to fragmented value */
529*4882a593Smuzhiyun 					/* bit 7 set -> required ea */
530*4882a593Smuzhiyun   u8 namelen;				/* length of name, bytes */
531*4882a593Smuzhiyun   u8 valuelen_lo;			/* length of value, bytes */
532*4882a593Smuzhiyun   u8 valuelen_hi;			/* length of value, bytes */
533*4882a593Smuzhiyun   u8 name[];
534*4882a593Smuzhiyun   /*
535*4882a593Smuzhiyun     u8 name[namelen];			ascii attrib name
536*4882a593Smuzhiyun     u8 nul;				terminating '\0', not counted
537*4882a593Smuzhiyun     u8 value[valuelen];			value, arbitrary
538*4882a593Smuzhiyun       if this.flags & 1, valuelen is 8 and the value is
539*4882a593Smuzhiyun         u32 length;			real length of value, bytes
540*4882a593Smuzhiyun         secno secno;			sector address where it starts
541*4882a593Smuzhiyun       if this.anode, the above sector number is the root of an anode tree
542*4882a593Smuzhiyun         which points to the value.
543*4882a593Smuzhiyun   */
544*4882a593Smuzhiyun };
545*4882a593Smuzhiyun 
ea_indirect(struct extended_attribute * ea)546*4882a593Smuzhiyun static inline bool ea_indirect(struct extended_attribute *ea)
547*4882a593Smuzhiyun {
548*4882a593Smuzhiyun 	return ea->flags & EA_indirect;
549*4882a593Smuzhiyun }
550*4882a593Smuzhiyun 
ea_in_anode(struct extended_attribute * ea)551*4882a593Smuzhiyun static inline bool ea_in_anode(struct extended_attribute *ea)
552*4882a593Smuzhiyun {
553*4882a593Smuzhiyun 	return ea->flags & EA_anode;
554*4882a593Smuzhiyun }
555*4882a593Smuzhiyun 
556*4882a593Smuzhiyun /*
557*4882a593Smuzhiyun    Local Variables:
558*4882a593Smuzhiyun    comment-column: 40
559*4882a593Smuzhiyun    End:
560*4882a593Smuzhiyun */
561