Subversion Repositories HelenOS

Rev

Rev 3497 | Rev 3506 | Go to most recent revision | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
2627 jermar 1
/*
2789 jermar 2
 * Copyright (c) 2008 Jakub Jermar
2627 jermar 3
 * All rights reserved.
4
 *
5
 * Redistribution and use in source and binary forms, with or without
6
 * modification, are permitted provided that the following conditions
7
 * are met:
8
 *
9
 * - Redistributions of source code must retain the above copyright
10
 *   notice, this list of conditions and the following disclaimer.
11
 * - Redistributions in binary form must reproduce the above copyright
12
 *   notice, this list of conditions and the following disclaimer in the
13
 *   documentation and/or other materials provided with the distribution.
14
 * - The name of the author may not be used to endorse or promote products
15
 *   derived from this software without specific prior written permission.
16
 *
17
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18
 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20
 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22
 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26
 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
 */
28
 
29
/** @addtogroup fs
30
 * @{
31
 */
32
 
33
#ifndef FAT_FAT_H_
34
#define FAT_FAT_H_
35
 
36
#include <ipc/ipc.h>
2643 jermar 37
#include <libfs.h>
2627 jermar 38
#include <atomic.h>
39
#include <sys/types.h>
40
#include <bool.h>
2789 jermar 41
#include "../../vfs/vfs.h"
2627 jermar 42
 
3111 jermar 43
#ifndef dprintf
2627 jermar 44
#define dprintf(...)    printf(__VA_ARGS__)
3111 jermar 45
#endif
2627 jermar 46
 
2628 jermar 47
typedef struct {
48
    uint8_t     ji[3];      /**< Jump instruction. */
49
    uint8_t     oem_name[8];
50
    /* BIOS Parameter Block */
51
    uint16_t    bps;        /**< Bytes per sector. */
52
    uint8_t     spc;        /**< Sectors per cluster. */
2789 jermar 53
    uint16_t    rscnt;      /**< Reserved sector count. */
2628 jermar 54
    uint8_t     fatcnt;     /**< Number of FATs. */
55
    uint16_t    root_ent_max;   /**< Maximum number of root directory
56
                         entries. */
2789 jermar 57
    uint16_t    totsec16;   /**< Total sectors. 16-bit version. */
2628 jermar 58
    uint8_t     mdesc;      /**< Media descriptor. */
59
    uint16_t    sec_per_fat;    /**< Sectors per FAT12/FAT16. */
60
    uint16_t    sec_per_track;  /**< Sectors per track. */
61
    uint16_t    headcnt;    /**< Number of heads. */
62
    uint32_t    hidden_sec; /**< Hidden sectors. */
2831 jermar 63
    uint32_t    totsec32;   /**< Total sectors. 32-bit version. */
2628 jermar 64
 
65
    union {
66
        struct {
67
            /* FAT12/FAT16 only: Extended BIOS Parameter Block */
68
            /** Physical drive number. */
69
            uint8_t     pdn;
70
            uint8_t     reserved;
71
            /** Extended boot signature. */
72
            uint8_t     ebs;
73
            /** Serial number. */
74
            uint32_t    id;
75
            /** Volume label. */
76
            uint8_t     label[11];
77
            /** FAT type. */
78
            uint8_t     type[8];
79
            /** Boot code. */
80
            uint8_t     boot_code[448];
81
            /** Boot sector signature. */
82
            uint16_t    signature;
83
        } __attribute__ ((packed));
84
        struct {
85
            /* FAT32 only */
86
            /** Sectors per FAT. */
87
            uint32_t    sectors_per_fat;
88
            /** FAT flags. */
89
            uint16_t    flags;
90
            /** Version. */
91
            uint16_t    version;
92
            /** Cluster number of root directory. */
93
            uint32_t    root_cluster;
94
            /** Sector number of file system information sector. */
95
            uint16_t    fsinfo_sec;
96
            /** Sector number of boot sector copy. */
97
            uint16_t    bscopy_sec;
98
            uint8_t     reserved1[12];
99
            /** Physical drive number. */
100
            uint8_t     pdn;
101
            uint8_t     reserved2;
102
            /** Extended boot signature. */
103
            uint8_t     ebs;
104
            /** Serial number. */
105
            uint32_t    id;
106
            /** Volume label. */
2789 jermar 107
            uint8_t     label[11];
2628 jermar 108
            /** FAT type. */
109
            uint8_t     type[8];
110
            /** Boot code. */
111
            uint8_t     boot_code[420];
112
            /** Signature. */
113
            uint16_t    signature;
114
        } __attribute__ ((packed));
115
    };
2629 jermar 116
} __attribute__ ((packed)) fat_bs_t;
2628 jermar 117
 
2864 jermar 118
typedef uint16_t fat_cluster_t;
119
 
2789 jermar 120
typedef enum {
2831 jermar 121
    FAT_INVALID,
2789 jermar 122
    FAT_DIRECTORY,
123
    FAT_FILE
124
} fat_node_type_t;
125
 
2864 jermar 126
struct fat_node;
127
 
128
/** FAT index structure.
129
 *
130
 * This structure exists to help us to overcome certain limitations of the FAT
131
 * file system design.  The problem with FAT is that it is hard to find
132
 * an entity which could represent a VFS index.  There are two candidates:
133
 *
134
 * a) number of the node's first cluster
135
 * b) the pair of the parent directory's first cluster and the dentry index
136
 *    within the parent directory
137
 *
138
 * We need VFS indices to be:
139
 * A) unique
140
 * B) stable in time, at least until the next mount
141
 *
142
 * Unfortunately a) does not meet the A) criterion because zero-length files
143
 * will have the first cluster field cleared.  And b) does not meet the B)
144
 * criterion because unlink() and rename() will both free up the original
145
 * dentry, which contains all the essential info about the file.
146
 *
147
 * Therefore, a completely opaque indices are used and the FAT server maintains
148
 * a mapping between them and otherwise nice b) variant.  On rename(), the VFS
149
 * index stays unaltered, while the internal FAT "physical tree address"
150
 * changes.  The unlink case is also handled this way thanks to an in-core node
151
 * pointer embedded in the index structure.
152
 */
153
typedef struct {
2890 jermar 154
    /** Used indices (position) hash table link. */
155
    link_t      uph_link;
156
    /** Used indices (index) hash table link. */
157
    link_t      uih_link;
2889 jermar 158
 
2951 jermar 159
    futex_t     lock;
2864 jermar 160
    dev_handle_t    dev_handle;
161
    fs_index_t  index;
162
    /**
2889 jermar 163
     * Parent node's first cluster.
2864 jermar 164
     * Zero is used if this node is not linked, in which case nodep must
165
     * contain a pointer to the in-core node structure.
166
     * One is used when the parent is the root directory.
167
     */
168
    fat_cluster_t   pfc;
2889 jermar 169
    /** Directory entry index within the parent node. */
2864 jermar 170
    unsigned    pdi;
171
    /** Pointer to in-core node instance. */
172
    struct fat_node *nodep;
173
} fat_idx_t;
174
 
2789 jermar 175
/** FAT in-core node. */
2864 jermar 176
typedef struct fat_node {
2951 jermar 177
    futex_t         lock;
2789 jermar 178
    fat_node_type_t     type;
2864 jermar 179
    fat_idx_t       *idx;
180
    /**
181
     *  Node's first cluster.
182
     *  Zero is used for zero-length nodes.
183
     *  One is used to mark root directory.
184
     */
185
    fat_cluster_t       firstc;
2831 jermar 186
    /** FAT in-core node free list link. */
187
    link_t          ffn_link;
2789 jermar 188
    size_t          size;
189
    unsigned        lnkcnt;
2831 jermar 190
    unsigned        refcnt;
191
    bool            dirty;
2789 jermar 192
} fat_node_t;
193
 
2643 jermar 194
extern fs_reg_t fat_reg;
2627 jermar 195
 
3110 jermar 196
extern void fat_mounted(ipc_callid_t, ipc_call_t *);
197
extern void fat_mount(ipc_callid_t, ipc_call_t *);
2638 jermar 198
extern void fat_lookup(ipc_callid_t, ipc_call_t *);
3307 jermar 199
extern void fat_read(ipc_callid_t, ipc_call_t *);
3497 jermar 200
extern void fat_write(ipc_callid_t, ipc_call_t *);
2638 jermar 201
 
2890 jermar 202
extern fat_idx_t *fat_idx_get_by_pos(dev_handle_t, fat_cluster_t, unsigned);
203
extern fat_idx_t *fat_idx_get_by_index(dev_handle_t, fs_index_t);
2876 jermar 204
 
3110 jermar 205
extern int fat_idx_init(void);
206
extern void fat_idx_fini(void);
207
extern int fat_idx_init_by_dev_handle(dev_handle_t);
208
extern void fat_idx_fini_by_dev_handle(dev_handle_t);
209
 
2627 jermar 210
#endif
211
 
212
/**
213
 * @}
214
 */