source: mainline/uspace/lib/ext4/libext4_superblock.c@ ae3d4f8

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since ae3d4f8 was ae3d4f8, checked in by Frantisek Princ <frantisek.princ@…>, 14 years ago

superblock update after block (de)allocation

  • Property mode set to 100644
File size: 8.7 KB
RevLine 
[eb91db7]1/*
2 * Copyright (c) 2011 Frantisek Princ
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 libext4
30 * @{
31 */
32
33/**
34 * @file libext4_superblock.c
[9875711]35 * @brief Ext4 superblock operations.
[eb91db7]36 */
37
[01ab41b]38#include <byteorder.h>
39#include <errno.h>
40#include <libblock.h>
41#include <malloc.h>
[3711e7e]42#include "libext4.h"
[eb91db7]43
[3712434]44uint32_t ext4_superblock_get_inodes_count(ext4_superblock_t *sb)
[01ab41b]45{
[3712434]46 return uint32_t_le2host(sb->inodes_count);
[01ab41b]47}
48
[3712434]49uint64_t ext4_superblock_get_blocks_count(ext4_superblock_t *sb)
50{
51 return ((uint64_t)uint32_t_le2host(sb->blocks_count_hi) << 32) |
52 uint32_t_le2host(sb->blocks_count_lo);
53}
54
55uint64_t ext4_superblock_get_reserved_blocks_count(ext4_superblock_t *sb)
56{
57 return ((uint64_t)uint32_t_le2host(sb->reserved_blocks_count_hi) << 32) |
58 uint32_t_le2host(sb->reserved_blocks_count_lo);
59}
60
61uint64_t ext4_superblock_get_free_blocks_count(ext4_superblock_t *sb)
62{
63 return ((uint64_t)uint32_t_le2host(sb->free_blocks_count_hi) << 32) |
64 uint32_t_le2host(sb->free_blocks_count_lo);
65}
66
[ae3d4f8]67void ext4_superblock_set_free_blocks_count(ext4_superblock_t *sb, uint64_t count)
68{
69 sb->free_blocks_count_lo = host2uint32_t_le((count << 32) >> 32);
70 sb->free_blocks_count_hi = host2uint32_t_le(count >> 32);
71}
72
[3712434]73uint32_t ext4_superblock_get_free_inodes_count(ext4_superblock_t *sb)
74{
75 return uint32_t_le2host(sb->free_inodes_count);
76}
77
78uint32_t ext4_superblock_get_first_data_block(ext4_superblock_t *sb)
[01ab41b]79{
[9c0c0e1]80 return uint32_t_le2host(sb->first_data_block);
[01ab41b]81}
82
[3712434]83uint32_t ext4_superblock_get_log_block_size(ext4_superblock_t *sb)
[01ab41b]84{
[9c0c0e1]85 return uint32_t_le2host(sb->log_block_size);
[01ab41b]86}
87
88uint32_t ext4_superblock_get_block_size(ext4_superblock_t *sb)
89{
[3712434]90 return 1024 << ext4_superblock_get_log_block_size(sb);
91}
92
93
94uint32_t ext4_superblock_get_blocks_per_group(ext4_superblock_t *sb)
95{
96 return uint32_t_le2host(sb->blocks_per_group);
97}
98
99uint32_t ext4_superblock_get_inodes_per_group(ext4_superblock_t *sb)
100{
101 return uint32_t_le2host(sb->inodes_per_group);
102}
103
104uint32_t ext4_superblock_get_mount_time(ext4_superblock_t *sb)
105{
106 return uint32_t_le2host(sb->mount_time);
107}
108
109uint32_t ext4_superblock_get_write_time(ext4_superblock_t *sb)
110{
111 return uint32_t_le2host(sb->write_time);
112}
113
114uint16_t ext4_superblock_get_mount_count(ext4_superblock_t *sb)
115{
116 return uint16_t_le2host(sb->mount_count);
117}
118
119uint16_t ext4_superblock_get_max_mount_count(ext4_superblock_t *sb)
120{
121 return uint16_t_le2host(sb->max_mount_count);
122}
123
124uint16_t ext4_superblock_get_magic(ext4_superblock_t *sb)
125{
126 return uint16_t_le2host(sb->magic);
127}
128
129uint16_t ext4_superblock_get_state(ext4_superblock_t *sb)
130{
131 return uint16_t_le2host(sb->state);
132}
133
134uint16_t ext4_superblock_get_errors(ext4_superblock_t *sb)
135{
136 return uint16_t_le2host(sb->errors);
137}
138
139
140uint16_t ext4_superblock_get_minor_rev_level(ext4_superblock_t *sb)
141{
142 return uint16_t_le2host(sb->minor_rev_level);
143}
144
145uint32_t ext4_superblock_get_last_check_time(ext4_superblock_t *sb)
146{
147 return uint32_t_le2host(sb->last_check_time);
148}
149
150uint32_t ext4_superblock_get_check_interval(ext4_superblock_t *sb){
151 return uint32_t_le2host(sb->check_interval);
152}
153
154uint32_t ext4_superblock_get_creator_os(ext4_superblock_t *sb)
155{
156 return uint32_t_le2host(sb->creator_os);
[01ab41b]157}
158
[9c0c0e1]159uint32_t ext4_superblock_get_rev_level(ext4_superblock_t *sb)
160{
161 return uint32_t_le2host(sb->rev_level);
162}
163
[3711e7e]164uint16_t ext4_superblock_get_inode_size(ext4_superblock_t *sb)
165{
166 if (ext4_superblock_get_rev_level(sb) == 0) {
167 return EXT4_REV0_INODE_SIZE;
168 }
169 return uint16_t_le2host(sb->inode_size);
170}
171
[3712434]172uint16_t ext4_superblock_get_block_group_number(ext4_superblock_t *sb)
[3711e7e]173{
[3712434]174 return uint16_t_le2host(sb->block_group_number);
[3711e7e]175}
176
[9c0c0e1]177uint32_t ext4_superblock_get_features_compatible(ext4_superblock_t *sb)
178{
179 return uint32_t_le2host(sb->features_compatible);
180}
181
182uint32_t ext4_superblock_get_features_incompatible(ext4_superblock_t *sb)
183{
184 return uint32_t_le2host(sb->features_incompatible);
185}
186
187uint32_t ext4_superblock_get_features_read_only(ext4_superblock_t *sb)
188{
189 return uint32_t_le2host(sb->features_read_only);
190}
191
[01ab41b]192
[7bc4508]193uint32_t* ext4_superblock_get_hash_seed(ext4_superblock_t *sb)
194{
195 return sb->hash_seed;
196}
197
[c25e39b]198uint16_t ext4_superblock_get_desc_size(ext4_superblock_t *sb)
199{
200 uint16_t size = uint16_t_le2host(sb->desc_size);
201
202 if (size < EXT4_BLOCK_MIN_GROUP_DESCRIPTOR_SIZE) {
203 size = EXT4_BLOCK_MIN_GROUP_DESCRIPTOR_SIZE;
204 }
205
206 return size;
207}
208
[7bc4508]209uint32_t ext4_superblock_get_flags(ext4_superblock_t *sb)
210{
211 return uint32_t_le2host(sb->flags);
212}
213
214
[3712434]215/*
[c25e39b]216 * More complex superblock operations
[01ab41b]217 */
[3712434]218
[7bc4508]219bool ext4_superblock_has_flag(ext4_superblock_t *sb, uint32_t flag)
220{
221 if (ext4_superblock_get_flags(sb) & flag) {
222 return true;
223 }
224 return false;
225}
226
[c25e39b]227// Feature checkers
228bool ext4_superblock_has_feature_compatible(ext4_superblock_t *sb, uint32_t feature)
229{
230 if (ext4_superblock_get_features_compatible(sb) & feature) {
231 return true;
232 }
233 return false;
234}
235
236bool ext4_superblock_has_feature_incompatible(ext4_superblock_t *sb, uint32_t feature)
237{
238 if (ext4_superblock_get_features_incompatible(sb) & feature) {
239 return true;
240 }
241 return false;
242}
243
244bool ext4_superblock_has_feature_read_only(ext4_superblock_t *sb, uint32_t feature)
245{
246 if (ext4_superblock_get_features_read_only(sb) & feature) {
247 return true;
248 }
249 return false;
250}
251
252
[01ab41b]253int ext4_superblock_read_direct(service_id_t service_id,
254 ext4_superblock_t **superblock)
255{
256 void *data;
257 int rc;
258
259 data = malloc(EXT4_SUPERBLOCK_SIZE);
260 if (data == NULL) {
261 return ENOMEM;
262 }
263
264 rc = block_read_bytes_direct(service_id, EXT4_SUPERBLOCK_OFFSET,
265 EXT4_SUPERBLOCK_SIZE, data);
266
267 if (rc != EOK) {
268 free(data);
269 return rc;
270 }
271
272 (*superblock) = data;
273
274 return EOK;
275}
276
[ae3d4f8]277int ext4_superblock_write_direct(service_id_t service_id,
278 ext4_superblock_t *sb)
279{
280 int rc;
281 uint32_t phys_block_size;
282 uint64_t first_block;
283 uint32_t block_count;
284
285 rc = block_get_bsize(service_id, &phys_block_size);
286 if (rc != EOK) {
287 // TODO error
288 return rc;
289 }
290
291 first_block = EXT4_SUPERBLOCK_OFFSET / phys_block_size;
292 block_count = EXT4_SUPERBLOCK_SIZE / phys_block_size;
293
294 if (EXT4_SUPERBLOCK_SIZE % phys_block_size) {
295 block_count++;
296 }
297
298 return block_write_direct(service_id, first_block, block_count, sb);
299
300}
301
302
[01ab41b]303int ext4_superblock_check_sanity(ext4_superblock_t *sb)
304{
[9c0c0e1]305 if (ext4_superblock_get_magic(sb) != EXT4_SUPERBLOCK_MAGIC) {
306 return ENOTSUP;
307 }
308
309 // TODO more checks !!!
310
[01ab41b]311 return EOK;
312}
[eb91db7]313
[b12ca16]314uint32_t ext4_superblock_get_block_group_count(ext4_superblock_t *sb)
315{
316 uint64_t blocks_count = ext4_superblock_get_blocks_count(sb);
317 uint32_t blocks_per_group = ext4_superblock_get_blocks_per_group(sb);
318
319 uint32_t block_groups_count = blocks_count / blocks_per_group;
320
321 if (blocks_count % blocks_per_group) {
322 block_groups_count++;
323 }
324
325 return block_groups_count;
326
327}
328
329uint32_t ext4_superblock_get_blocks_in_group(ext4_superblock_t *sb, uint32_t bgid)
330{
331 uint32_t block_group_count = ext4_superblock_get_block_group_count(sb);
332 uint32_t blocks_per_group = ext4_superblock_get_blocks_per_group(sb);
333 uint64_t total_blocks = ext4_superblock_get_blocks_count(sb);
334
335 if (bgid < block_group_count - 1) {
336 return blocks_per_group;
337 } else {
338 return (total_blocks - ((block_group_count - 1) * blocks_per_group));
339 }
340
341}
342
343uint32_t ext4_superblock_get_inodes_in_group(ext4_superblock_t *sb, uint32_t bgid)
344{
345 uint32_t block_group_count = ext4_superblock_get_block_group_count(sb);
346 uint32_t inodes_per_group = ext4_superblock_get_inodes_per_group(sb);
347 uint32_t total_inodes = ext4_superblock_get_inodes_count(sb);
348
349 if (bgid < block_group_count - 1) {
350 return inodes_per_group;
351 } else {
352 return (total_inodes - ((block_group_count - 1) * inodes_per_group));
353 }
354
355}
356
[eb91db7]357/**
358 * @}
359 */
Note: See TracBrowser for help on using the repository browser.