ext4_write.c 29 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * (C) Copyright 2011 - 2012 Samsung Electronics
  4. * EXT4 filesystem implementation in Uboot by
  5. * Uma Shankar <uma.shankar@samsung.com>
  6. * Manjunatha C Achar <a.manjunatha@samsung.com>
  7. *
  8. * ext4ls and ext4load : Based on ext2 ls and load support in Uboot.
  9. * Ext4 read optimization taken from Open-Moko
  10. * Qi bootloader
  11. *
  12. * (C) Copyright 2004
  13. * esd gmbh <www.esd-electronics.com>
  14. * Reinhard Arlt <reinhard.arlt@esd-electronics.com>
  15. *
  16. * based on code from grub2 fs/ext2.c and fs/fshelp.c by
  17. * GRUB -- GRand Unified Bootloader
  18. * Copyright (C) 2003, 2004 Free Software Foundation, Inc.
  19. *
  20. * ext4write : Based on generic ext4 protocol.
  21. */
  22. #include <common.h>
  23. #include <blk.h>
  24. #include <log.h>
  25. #include <malloc.h>
  26. #include <memalign.h>
  27. #include <part.h>
  28. #include <linux/stat.h>
  29. #include <div64.h>
  30. #include "ext4_common.h"
  31. static inline void ext4fs_sb_free_inodes_inc(struct ext2_sblock *sb)
  32. {
  33. sb->free_inodes = cpu_to_le32(le32_to_cpu(sb->free_inodes) + 1);
  34. }
  35. static inline void ext4fs_sb_free_blocks_inc(struct ext2_sblock *sb)
  36. {
  37. sb->free_blocks = cpu_to_le32(le32_to_cpu(sb->free_blocks) + 1);
  38. }
  39. static inline void ext4fs_bg_free_inodes_inc
  40. (struct ext2_block_group *bg, const struct ext_filesystem *fs)
  41. {
  42. uint32_t free_inodes = le16_to_cpu(bg->free_inodes);
  43. if (fs->gdsize == 64)
  44. free_inodes += le16_to_cpu(bg->free_inodes_high) << 16;
  45. free_inodes++;
  46. bg->free_inodes = cpu_to_le16(free_inodes & 0xffff);
  47. if (fs->gdsize == 64)
  48. bg->free_inodes_high = cpu_to_le16(free_inodes >> 16);
  49. }
  50. static inline void ext4fs_bg_free_blocks_inc
  51. (struct ext2_block_group *bg, const struct ext_filesystem *fs)
  52. {
  53. uint32_t free_blocks = le16_to_cpu(bg->free_blocks);
  54. if (fs->gdsize == 64)
  55. free_blocks += le16_to_cpu(bg->free_blocks_high) << 16;
  56. free_blocks++;
  57. bg->free_blocks = cpu_to_le16(free_blocks & 0xffff);
  58. if (fs->gdsize == 64)
  59. bg->free_blocks_high = cpu_to_le16(free_blocks >> 16);
  60. }
  61. static void ext4fs_update(void)
  62. {
  63. short i;
  64. ext4fs_update_journal();
  65. struct ext_filesystem *fs = get_fs();
  66. struct ext2_block_group *bgd = NULL;
  67. /* update super block */
  68. put_ext4((uint64_t)(SUPERBLOCK_SIZE),
  69. (struct ext2_sblock *)fs->sb, (uint32_t)SUPERBLOCK_SIZE);
  70. /* update block bitmaps */
  71. for (i = 0; i < fs->no_blkgrp; i++) {
  72. bgd = ext4fs_get_group_descriptor(fs, i);
  73. bgd->bg_checksum = cpu_to_le16(ext4fs_checksum_update(i));
  74. uint64_t b_bitmap_blk = ext4fs_bg_get_block_id(bgd, fs);
  75. put_ext4(b_bitmap_blk * fs->blksz,
  76. fs->blk_bmaps[i], fs->blksz);
  77. }
  78. /* update inode bitmaps */
  79. for (i = 0; i < fs->no_blkgrp; i++) {
  80. bgd = ext4fs_get_group_descriptor(fs, i);
  81. uint64_t i_bitmap_blk = ext4fs_bg_get_inode_id(bgd, fs);
  82. put_ext4(i_bitmap_blk * fs->blksz,
  83. fs->inode_bmaps[i], fs->blksz);
  84. }
  85. /* update the block group descriptor table */
  86. put_ext4((uint64_t)((uint64_t)fs->gdtable_blkno * (uint64_t)fs->blksz),
  87. (struct ext2_block_group *)fs->gdtable,
  88. (fs->blksz * fs->no_blk_pergdt));
  89. ext4fs_dump_metadata();
  90. gindex = 0;
  91. gd_index = 0;
  92. }
  93. int ext4fs_get_bgdtable(void)
  94. {
  95. int status;
  96. struct ext_filesystem *fs = get_fs();
  97. int gdsize_total = ROUND(fs->no_blkgrp * fs->gdsize, fs->blksz);
  98. fs->no_blk_pergdt = gdsize_total / fs->blksz;
  99. /* allocate memory for gdtable */
  100. fs->gdtable = zalloc(gdsize_total);
  101. if (!fs->gdtable)
  102. return -ENOMEM;
  103. /* read the group descriptor table */
  104. status = ext4fs_devread((lbaint_t)fs->gdtable_blkno * fs->sect_perblk,
  105. 0, fs->blksz * fs->no_blk_pergdt, fs->gdtable);
  106. if (status == 0)
  107. goto fail;
  108. if (ext4fs_log_gdt(fs->gdtable)) {
  109. printf("Error in ext4fs_log_gdt\n");
  110. return -1;
  111. }
  112. return 0;
  113. fail:
  114. free(fs->gdtable);
  115. fs->gdtable = NULL;
  116. return -1;
  117. }
  118. static void delete_single_indirect_block(struct ext2_inode *inode)
  119. {
  120. struct ext2_block_group *bgd = NULL;
  121. static int prev_bg_bmap_idx = -1;
  122. uint32_t blknr;
  123. int remainder;
  124. int bg_idx;
  125. int status;
  126. uint32_t blk_per_grp = le32_to_cpu(ext4fs_root->sblock.blocks_per_group);
  127. struct ext_filesystem *fs = get_fs();
  128. char *journal_buffer = zalloc(fs->blksz);
  129. if (!journal_buffer) {
  130. printf("No memory\n");
  131. return;
  132. }
  133. /* deleting the single indirect block associated with inode */
  134. if (inode->b.blocks.indir_block != 0) {
  135. blknr = le32_to_cpu(inode->b.blocks.indir_block);
  136. debug("SIPB releasing %u\n", blknr);
  137. bg_idx = blknr / blk_per_grp;
  138. if (fs->blksz == 1024) {
  139. remainder = blknr % blk_per_grp;
  140. if (!remainder)
  141. bg_idx--;
  142. }
  143. ext4fs_reset_block_bmap(blknr, fs->blk_bmaps[bg_idx], bg_idx);
  144. /* get block group descriptor table */
  145. bgd = ext4fs_get_group_descriptor(fs, bg_idx);
  146. ext4fs_bg_free_blocks_inc(bgd, fs);
  147. ext4fs_sb_free_blocks_inc(fs->sb);
  148. /* journal backup */
  149. if (prev_bg_bmap_idx != bg_idx) {
  150. uint64_t b_bitmap_blk = ext4fs_bg_get_block_id(bgd, fs);
  151. status = ext4fs_devread(
  152. b_bitmap_blk * fs->sect_perblk,
  153. 0, fs->blksz, journal_buffer);
  154. if (status == 0)
  155. goto fail;
  156. if (ext4fs_log_journal(journal_buffer, b_bitmap_blk))
  157. goto fail;
  158. prev_bg_bmap_idx = bg_idx;
  159. }
  160. }
  161. fail:
  162. free(journal_buffer);
  163. }
  164. static void delete_double_indirect_block(struct ext2_inode *inode)
  165. {
  166. int i;
  167. short status;
  168. static int prev_bg_bmap_idx = -1;
  169. uint32_t blknr;
  170. int remainder;
  171. int bg_idx;
  172. uint32_t blk_per_grp = le32_to_cpu(ext4fs_root->sblock.blocks_per_group);
  173. __le32 *di_buffer = NULL;
  174. void *dib_start_addr = NULL;
  175. struct ext2_block_group *bgd = NULL;
  176. struct ext_filesystem *fs = get_fs();
  177. char *journal_buffer = zalloc(fs->blksz);
  178. if (!journal_buffer) {
  179. printf("No memory\n");
  180. return;
  181. }
  182. if (inode->b.blocks.double_indir_block != 0) {
  183. di_buffer = zalloc(fs->blksz);
  184. if (!di_buffer) {
  185. printf("No memory\n");
  186. return;
  187. }
  188. dib_start_addr = di_buffer;
  189. blknr = le32_to_cpu(inode->b.blocks.double_indir_block);
  190. status = ext4fs_devread((lbaint_t)blknr * fs->sect_perblk, 0,
  191. fs->blksz, (char *)di_buffer);
  192. for (i = 0; i < fs->blksz / sizeof(int); i++) {
  193. if (*di_buffer == 0)
  194. break;
  195. debug("DICB releasing %u\n", *di_buffer);
  196. bg_idx = le32_to_cpu(*di_buffer) / blk_per_grp;
  197. if (fs->blksz == 1024) {
  198. remainder = le32_to_cpu(*di_buffer) % blk_per_grp;
  199. if (!remainder)
  200. bg_idx--;
  201. }
  202. /* get block group descriptor table */
  203. bgd = ext4fs_get_group_descriptor(fs, bg_idx);
  204. ext4fs_reset_block_bmap(le32_to_cpu(*di_buffer),
  205. fs->blk_bmaps[bg_idx], bg_idx);
  206. di_buffer++;
  207. ext4fs_bg_free_blocks_inc(bgd, fs);
  208. ext4fs_sb_free_blocks_inc(fs->sb);
  209. /* journal backup */
  210. if (prev_bg_bmap_idx != bg_idx) {
  211. uint64_t b_bitmap_blk =
  212. ext4fs_bg_get_block_id(bgd, fs);
  213. status = ext4fs_devread(b_bitmap_blk
  214. * fs->sect_perblk, 0,
  215. fs->blksz,
  216. journal_buffer);
  217. if (status == 0)
  218. goto fail;
  219. if (ext4fs_log_journal(journal_buffer,
  220. b_bitmap_blk))
  221. goto fail;
  222. prev_bg_bmap_idx = bg_idx;
  223. }
  224. }
  225. /* removing the parent double indirect block */
  226. blknr = le32_to_cpu(inode->b.blocks.double_indir_block);
  227. bg_idx = blknr / blk_per_grp;
  228. if (fs->blksz == 1024) {
  229. remainder = blknr % blk_per_grp;
  230. if (!remainder)
  231. bg_idx--;
  232. }
  233. /* get block group descriptor table */
  234. bgd = ext4fs_get_group_descriptor(fs, bg_idx);
  235. ext4fs_reset_block_bmap(blknr, fs->blk_bmaps[bg_idx], bg_idx);
  236. ext4fs_bg_free_blocks_inc(bgd, fs);
  237. ext4fs_sb_free_blocks_inc(fs->sb);
  238. /* journal backup */
  239. if (prev_bg_bmap_idx != bg_idx) {
  240. uint64_t b_bitmap_blk = ext4fs_bg_get_block_id(bgd, fs);
  241. status = ext4fs_devread(b_bitmap_blk * fs->sect_perblk,
  242. 0, fs->blksz, journal_buffer);
  243. if (status == 0)
  244. goto fail;
  245. if (ext4fs_log_journal(journal_buffer, b_bitmap_blk))
  246. goto fail;
  247. prev_bg_bmap_idx = bg_idx;
  248. }
  249. debug("DIPB releasing %d\n", blknr);
  250. }
  251. fail:
  252. free(dib_start_addr);
  253. free(journal_buffer);
  254. }
  255. static void delete_triple_indirect_block(struct ext2_inode *inode)
  256. {
  257. int i, j;
  258. short status;
  259. static int prev_bg_bmap_idx = -1;
  260. uint32_t blknr;
  261. int remainder;
  262. int bg_idx;
  263. uint32_t blk_per_grp = le32_to_cpu(ext4fs_root->sblock.blocks_per_group);
  264. __le32 *tigp_buffer = NULL;
  265. void *tib_start_addr = NULL;
  266. __le32 *tip_buffer = NULL;
  267. void *tipb_start_addr = NULL;
  268. struct ext2_block_group *bgd = NULL;
  269. struct ext_filesystem *fs = get_fs();
  270. char *journal_buffer = zalloc(fs->blksz);
  271. if (!journal_buffer) {
  272. printf("No memory\n");
  273. return;
  274. }
  275. if (inode->b.blocks.triple_indir_block != 0) {
  276. tigp_buffer = zalloc(fs->blksz);
  277. if (!tigp_buffer) {
  278. printf("No memory\n");
  279. return;
  280. }
  281. tib_start_addr = tigp_buffer;
  282. blknr = le32_to_cpu(inode->b.blocks.triple_indir_block);
  283. status = ext4fs_devread((lbaint_t)blknr * fs->sect_perblk, 0,
  284. fs->blksz, (char *)tigp_buffer);
  285. for (i = 0; i < fs->blksz / sizeof(int); i++) {
  286. if (*tigp_buffer == 0)
  287. break;
  288. debug("tigp buffer releasing %u\n", *tigp_buffer);
  289. tip_buffer = zalloc(fs->blksz);
  290. if (!tip_buffer)
  291. goto fail;
  292. tipb_start_addr = tip_buffer;
  293. status = ext4fs_devread((lbaint_t)le32_to_cpu(*tigp_buffer) *
  294. fs->sect_perblk, 0, fs->blksz,
  295. (char *)tip_buffer);
  296. for (j = 0; j < fs->blksz / sizeof(int); j++) {
  297. if (le32_to_cpu(*tip_buffer) == 0)
  298. break;
  299. bg_idx = le32_to_cpu(*tip_buffer) / blk_per_grp;
  300. if (fs->blksz == 1024) {
  301. remainder = le32_to_cpu(*tip_buffer) % blk_per_grp;
  302. if (!remainder)
  303. bg_idx--;
  304. }
  305. ext4fs_reset_block_bmap(le32_to_cpu(*tip_buffer),
  306. fs->blk_bmaps[bg_idx],
  307. bg_idx);
  308. tip_buffer++;
  309. /* get block group descriptor table */
  310. bgd = ext4fs_get_group_descriptor(fs, bg_idx);
  311. ext4fs_bg_free_blocks_inc(bgd, fs);
  312. ext4fs_sb_free_blocks_inc(fs->sb);
  313. /* journal backup */
  314. if (prev_bg_bmap_idx != bg_idx) {
  315. uint64_t b_bitmap_blk =
  316. ext4fs_bg_get_block_id(bgd, fs);
  317. status =
  318. ext4fs_devread(
  319. b_bitmap_blk *
  320. fs->sect_perblk, 0,
  321. fs->blksz,
  322. journal_buffer);
  323. if (status == 0)
  324. goto fail;
  325. if (ext4fs_log_journal(journal_buffer,
  326. b_bitmap_blk))
  327. goto fail;
  328. prev_bg_bmap_idx = bg_idx;
  329. }
  330. }
  331. free(tipb_start_addr);
  332. tipb_start_addr = NULL;
  333. /*
  334. * removing the grand parent blocks
  335. * which is connected to inode
  336. */
  337. bg_idx = le32_to_cpu(*tigp_buffer) / blk_per_grp;
  338. if (fs->blksz == 1024) {
  339. remainder = le32_to_cpu(*tigp_buffer) % blk_per_grp;
  340. if (!remainder)
  341. bg_idx--;
  342. }
  343. ext4fs_reset_block_bmap(le32_to_cpu(*tigp_buffer),
  344. fs->blk_bmaps[bg_idx], bg_idx);
  345. tigp_buffer++;
  346. /* get block group descriptor table */
  347. bgd = ext4fs_get_group_descriptor(fs, bg_idx);
  348. ext4fs_bg_free_blocks_inc(bgd, fs);
  349. ext4fs_sb_free_blocks_inc(fs->sb);
  350. /* journal backup */
  351. if (prev_bg_bmap_idx != bg_idx) {
  352. uint64_t b_bitmap_blk =
  353. ext4fs_bg_get_block_id(bgd, fs);
  354. memset(journal_buffer, '\0', fs->blksz);
  355. status = ext4fs_devread(b_bitmap_blk *
  356. fs->sect_perblk, 0,
  357. fs->blksz,
  358. journal_buffer);
  359. if (status == 0)
  360. goto fail;
  361. if (ext4fs_log_journal(journal_buffer,
  362. b_bitmap_blk))
  363. goto fail;
  364. prev_bg_bmap_idx = bg_idx;
  365. }
  366. }
  367. /* removing the grand parent triple indirect block */
  368. blknr = le32_to_cpu(inode->b.blocks.triple_indir_block);
  369. bg_idx = blknr / blk_per_grp;
  370. if (fs->blksz == 1024) {
  371. remainder = blknr % blk_per_grp;
  372. if (!remainder)
  373. bg_idx--;
  374. }
  375. ext4fs_reset_block_bmap(blknr, fs->blk_bmaps[bg_idx], bg_idx);
  376. /* get block group descriptor table */
  377. bgd = ext4fs_get_group_descriptor(fs, bg_idx);
  378. ext4fs_bg_free_blocks_inc(bgd, fs);
  379. ext4fs_sb_free_blocks_inc(fs->sb);
  380. /* journal backup */
  381. if (prev_bg_bmap_idx != bg_idx) {
  382. uint64_t b_bitmap_blk = ext4fs_bg_get_block_id(bgd, fs);
  383. status = ext4fs_devread(b_bitmap_blk * fs->sect_perblk,
  384. 0, fs->blksz, journal_buffer);
  385. if (status == 0)
  386. goto fail;
  387. if (ext4fs_log_journal(journal_buffer, b_bitmap_blk))
  388. goto fail;
  389. prev_bg_bmap_idx = bg_idx;
  390. }
  391. debug("tigp buffer itself releasing %d\n", blknr);
  392. }
  393. fail:
  394. free(tib_start_addr);
  395. free(tipb_start_addr);
  396. free(journal_buffer);
  397. }
  398. static int ext4fs_delete_file(int inodeno)
  399. {
  400. struct ext2_inode inode;
  401. short status;
  402. int i;
  403. int remainder;
  404. long int blknr;
  405. int bg_idx;
  406. int ibmap_idx;
  407. char *read_buffer = NULL;
  408. char *start_block_address = NULL;
  409. uint32_t no_blocks;
  410. static int prev_bg_bmap_idx = -1;
  411. unsigned int inodes_per_block;
  412. uint32_t blkno;
  413. unsigned int blkoff;
  414. uint32_t blk_per_grp = le32_to_cpu(ext4fs_root->sblock.blocks_per_group);
  415. uint32_t inode_per_grp = le32_to_cpu(ext4fs_root->sblock.inodes_per_group);
  416. struct ext2_inode *inode_buffer = NULL;
  417. struct ext2_block_group *bgd = NULL;
  418. struct ext_filesystem *fs = get_fs();
  419. char *journal_buffer = zalloc(fs->blksz);
  420. if (!journal_buffer)
  421. return -ENOMEM;
  422. status = ext4fs_read_inode(ext4fs_root, inodeno, &inode);
  423. if (status == 0)
  424. goto fail;
  425. /* read the block no allocated to a file */
  426. no_blocks = le32_to_cpu(inode.size) / fs->blksz;
  427. if (le32_to_cpu(inode.size) % fs->blksz)
  428. no_blocks++;
  429. /*
  430. * special case for symlinks whose target are small enough that
  431. *it fits in struct ext2_inode.b.symlink: no block had been allocated
  432. */
  433. if (S_ISLNK(le16_to_cpu(inode.mode)) &&
  434. le32_to_cpu(inode.size) <= sizeof(inode.b.symlink)) {
  435. no_blocks = 0;
  436. }
  437. if (le32_to_cpu(inode.flags) & EXT4_EXTENTS_FL) {
  438. /* FIXME delete extent index blocks, i.e. eh_depth >= 1 */
  439. struct ext4_extent_header *eh =
  440. (struct ext4_extent_header *)
  441. inode.b.blocks.dir_blocks;
  442. debug("del: dep=%d entries=%d\n", eh->eh_depth, eh->eh_entries);
  443. } else {
  444. delete_single_indirect_block(&inode);
  445. delete_double_indirect_block(&inode);
  446. delete_triple_indirect_block(&inode);
  447. }
  448. /* release data blocks */
  449. for (i = 0; i < no_blocks; i++) {
  450. blknr = read_allocated_block(&inode, i, NULL);
  451. if (blknr == 0)
  452. continue;
  453. if (blknr < 0)
  454. goto fail;
  455. bg_idx = blknr / blk_per_grp;
  456. if (fs->blksz == 1024) {
  457. remainder = blknr % blk_per_grp;
  458. if (!remainder)
  459. bg_idx--;
  460. }
  461. ext4fs_reset_block_bmap(blknr, fs->blk_bmaps[bg_idx],
  462. bg_idx);
  463. debug("EXT4 Block releasing %ld: %d\n", blknr, bg_idx);
  464. /* get block group descriptor table */
  465. bgd = ext4fs_get_group_descriptor(fs, bg_idx);
  466. ext4fs_bg_free_blocks_inc(bgd, fs);
  467. ext4fs_sb_free_blocks_inc(fs->sb);
  468. /* journal backup */
  469. if (prev_bg_bmap_idx != bg_idx) {
  470. uint64_t b_bitmap_blk = ext4fs_bg_get_block_id(bgd, fs);
  471. status = ext4fs_devread(b_bitmap_blk * fs->sect_perblk,
  472. 0, fs->blksz,
  473. journal_buffer);
  474. if (status == 0)
  475. goto fail;
  476. if (ext4fs_log_journal(journal_buffer, b_bitmap_blk))
  477. goto fail;
  478. prev_bg_bmap_idx = bg_idx;
  479. }
  480. }
  481. /* release inode */
  482. /* from the inode no to blockno */
  483. inodes_per_block = fs->blksz / fs->inodesz;
  484. ibmap_idx = inodeno / inode_per_grp;
  485. /* get the block no */
  486. inodeno--;
  487. /* get block group descriptor table */
  488. bgd = ext4fs_get_group_descriptor(fs, ibmap_idx);
  489. blkno = ext4fs_bg_get_inode_table_id(bgd, fs) +
  490. (inodeno % inode_per_grp) / inodes_per_block;
  491. /* get the offset of the inode */
  492. blkoff = ((inodeno) % inodes_per_block) * fs->inodesz;
  493. /* read the block no containing the inode */
  494. read_buffer = zalloc(fs->blksz);
  495. if (!read_buffer)
  496. goto fail;
  497. start_block_address = read_buffer;
  498. status = ext4fs_devread((lbaint_t)blkno * fs->sect_perblk,
  499. 0, fs->blksz, read_buffer);
  500. if (status == 0)
  501. goto fail;
  502. if (ext4fs_log_journal(read_buffer, blkno))
  503. goto fail;
  504. read_buffer = read_buffer + blkoff;
  505. inode_buffer = (struct ext2_inode *)read_buffer;
  506. memset(inode_buffer, '\0', fs->inodesz);
  507. /* write the inode to original position in inode table */
  508. if (ext4fs_put_metadata(start_block_address, blkno))
  509. goto fail;
  510. /* update the respective inode bitmaps */
  511. inodeno++;
  512. ext4fs_reset_inode_bmap(inodeno, fs->inode_bmaps[ibmap_idx], ibmap_idx);
  513. ext4fs_bg_free_inodes_inc(bgd, fs);
  514. ext4fs_sb_free_inodes_inc(fs->sb);
  515. /* journal backup */
  516. memset(journal_buffer, '\0', fs->blksz);
  517. status = ext4fs_devread(ext4fs_bg_get_inode_id(bgd, fs) *
  518. fs->sect_perblk, 0, fs->blksz, journal_buffer);
  519. if (status == 0)
  520. goto fail;
  521. if (ext4fs_log_journal(journal_buffer, ext4fs_bg_get_inode_id(bgd, fs)))
  522. goto fail;
  523. ext4fs_update();
  524. ext4fs_deinit();
  525. ext4fs_reinit_global();
  526. if (ext4fs_init() != 0) {
  527. printf("error in File System init\n");
  528. goto fail;
  529. }
  530. free(start_block_address);
  531. free(journal_buffer);
  532. return 0;
  533. fail:
  534. free(start_block_address);
  535. free(journal_buffer);
  536. return -1;
  537. }
  538. int ext4fs_init(void)
  539. {
  540. short status;
  541. int i;
  542. uint32_t real_free_blocks = 0;
  543. struct ext_filesystem *fs = get_fs();
  544. /* populate fs */
  545. fs->blksz = EXT2_BLOCK_SIZE(ext4fs_root);
  546. fs->sect_perblk = fs->blksz >> fs->dev_desc->log2blksz;
  547. /* get the superblock */
  548. fs->sb = zalloc(SUPERBLOCK_SIZE);
  549. if (!fs->sb)
  550. return -ENOMEM;
  551. if (!ext4_read_superblock((char *)fs->sb))
  552. goto fail;
  553. /* init journal */
  554. if (ext4fs_init_journal())
  555. goto fail;
  556. /* get total no of blockgroups */
  557. fs->no_blkgrp = (uint32_t)ext4fs_div_roundup(
  558. le32_to_cpu(ext4fs_root->sblock.total_blocks)
  559. - le32_to_cpu(ext4fs_root->sblock.first_data_block),
  560. le32_to_cpu(ext4fs_root->sblock.blocks_per_group));
  561. /* get the block group descriptor table */
  562. fs->gdtable_blkno = ((EXT2_MIN_BLOCK_SIZE == fs->blksz) + 1);
  563. if (ext4fs_get_bgdtable() == -1) {
  564. printf("Error in getting the block group descriptor table\n");
  565. goto fail;
  566. }
  567. /* load all the available bitmap block of the partition */
  568. fs->blk_bmaps = zalloc(fs->no_blkgrp * sizeof(char *));
  569. if (!fs->blk_bmaps)
  570. goto fail;
  571. for (i = 0; i < fs->no_blkgrp; i++) {
  572. fs->blk_bmaps[i] = zalloc(fs->blksz);
  573. if (!fs->blk_bmaps[i])
  574. goto fail;
  575. }
  576. for (i = 0; i < fs->no_blkgrp; i++) {
  577. struct ext2_block_group *bgd =
  578. ext4fs_get_group_descriptor(fs, i);
  579. status = ext4fs_devread(ext4fs_bg_get_block_id(bgd, fs) *
  580. fs->sect_perblk, 0,
  581. fs->blksz, (char *)fs->blk_bmaps[i]);
  582. if (status == 0)
  583. goto fail;
  584. }
  585. /* load all the available inode bitmap of the partition */
  586. fs->inode_bmaps = zalloc(fs->no_blkgrp * sizeof(unsigned char *));
  587. if (!fs->inode_bmaps)
  588. goto fail;
  589. for (i = 0; i < fs->no_blkgrp; i++) {
  590. fs->inode_bmaps[i] = zalloc(fs->blksz);
  591. if (!fs->inode_bmaps[i])
  592. goto fail;
  593. }
  594. for (i = 0; i < fs->no_blkgrp; i++) {
  595. struct ext2_block_group *bgd =
  596. ext4fs_get_group_descriptor(fs, i);
  597. status = ext4fs_devread(ext4fs_bg_get_inode_id(bgd, fs) *
  598. fs->sect_perblk,
  599. 0, fs->blksz,
  600. (char *)fs->inode_bmaps[i]);
  601. if (status == 0)
  602. goto fail;
  603. }
  604. /*
  605. * check filesystem consistency with free blocks of file system
  606. * some time we observed that superblock freeblocks does not match
  607. * with the blockgroups freeblocks when improper
  608. * reboot of a linux kernel
  609. */
  610. for (i = 0; i < fs->no_blkgrp; i++) {
  611. struct ext2_block_group *bgd =
  612. ext4fs_get_group_descriptor(fs, i);
  613. real_free_blocks = real_free_blocks +
  614. ext4fs_bg_get_free_blocks(bgd, fs);
  615. }
  616. if (real_free_blocks != ext4fs_sb_get_free_blocks(fs->sb))
  617. ext4fs_sb_set_free_blocks(fs->sb, real_free_blocks);
  618. return 0;
  619. fail:
  620. ext4fs_deinit();
  621. return -1;
  622. }
  623. void ext4fs_deinit(void)
  624. {
  625. int i;
  626. struct ext2_inode inode_journal;
  627. struct journal_superblock_t *jsb;
  628. uint32_t blknr;
  629. struct ext_filesystem *fs = get_fs();
  630. uint32_t new_feature_incompat;
  631. /* free journal */
  632. char *temp_buff = zalloc(fs->blksz);
  633. if (temp_buff) {
  634. ext4fs_read_inode(ext4fs_root, EXT2_JOURNAL_INO,
  635. &inode_journal);
  636. blknr = read_allocated_block(&inode_journal,
  637. EXT2_JOURNAL_SUPERBLOCK, NULL);
  638. ext4fs_devread((lbaint_t)blknr * fs->sect_perblk, 0, fs->blksz,
  639. temp_buff);
  640. jsb = (struct journal_superblock_t *)temp_buff;
  641. jsb->s_start = 0;
  642. put_ext4((uint64_t) ((uint64_t)blknr * (uint64_t)fs->blksz),
  643. (struct journal_superblock_t *)temp_buff, fs->blksz);
  644. free(temp_buff);
  645. }
  646. ext4fs_free_journal();
  647. /* get the superblock */
  648. ext4_read_superblock((char *)fs->sb);
  649. new_feature_incompat = le32_to_cpu(fs->sb->feature_incompat);
  650. new_feature_incompat &= ~EXT3_FEATURE_INCOMPAT_RECOVER;
  651. fs->sb->feature_incompat = cpu_to_le32(new_feature_incompat);
  652. put_ext4((uint64_t)(SUPERBLOCK_SIZE),
  653. (struct ext2_sblock *)fs->sb, (uint32_t)SUPERBLOCK_SIZE);
  654. free(fs->sb);
  655. fs->sb = NULL;
  656. if (fs->blk_bmaps) {
  657. for (i = 0; i < fs->no_blkgrp; i++) {
  658. free(fs->blk_bmaps[i]);
  659. fs->blk_bmaps[i] = NULL;
  660. }
  661. free(fs->blk_bmaps);
  662. fs->blk_bmaps = NULL;
  663. }
  664. if (fs->inode_bmaps) {
  665. for (i = 0; i < fs->no_blkgrp; i++) {
  666. free(fs->inode_bmaps[i]);
  667. fs->inode_bmaps[i] = NULL;
  668. }
  669. free(fs->inode_bmaps);
  670. fs->inode_bmaps = NULL;
  671. }
  672. free(fs->gdtable);
  673. fs->gdtable = NULL;
  674. /*
  675. * reinitiliazed the global inode and
  676. * block bitmap first execution check variables
  677. */
  678. fs->first_pass_ibmap = 0;
  679. fs->first_pass_bbmap = 0;
  680. fs->curr_inode_no = 0;
  681. fs->curr_blkno = 0;
  682. }
  683. /*
  684. * Write data to filesystem blocks. Uses same optimization for
  685. * contigous sectors as ext4fs_read_file
  686. */
  687. static int ext4fs_write_file(struct ext2_inode *file_inode,
  688. int pos, unsigned int len, const char *buf)
  689. {
  690. int i;
  691. int blockcnt;
  692. uint32_t filesize = le32_to_cpu(file_inode->size);
  693. struct ext_filesystem *fs = get_fs();
  694. int log2blksz = fs->dev_desc->log2blksz;
  695. int log2_fs_blocksize = LOG2_BLOCK_SIZE(ext4fs_root) - log2blksz;
  696. int previous_block_number = -1;
  697. int delayed_start = 0;
  698. int delayed_extent = 0;
  699. int delayed_next = 0;
  700. const char *delayed_buf = NULL;
  701. /* Adjust len so it we can't read past the end of the file. */
  702. if (len > filesize)
  703. len = filesize;
  704. blockcnt = ((len + pos) + fs->blksz - 1) / fs->blksz;
  705. for (i = pos / fs->blksz; i < blockcnt; i++) {
  706. long int blknr;
  707. int blockend = fs->blksz;
  708. int skipfirst = 0;
  709. blknr = read_allocated_block(file_inode, i, NULL);
  710. if (blknr <= 0)
  711. return -1;
  712. blknr = blknr << log2_fs_blocksize;
  713. if (blknr) {
  714. if (previous_block_number != -1) {
  715. if (delayed_next == blknr) {
  716. delayed_extent += blockend;
  717. delayed_next += blockend >> log2blksz;
  718. } else { /* spill */
  719. put_ext4((uint64_t)
  720. ((uint64_t)delayed_start << log2blksz),
  721. delayed_buf,
  722. (uint32_t) delayed_extent);
  723. previous_block_number = blknr;
  724. delayed_start = blknr;
  725. delayed_extent = blockend;
  726. delayed_buf = buf;
  727. delayed_next = blknr +
  728. (blockend >> log2blksz);
  729. }
  730. } else {
  731. previous_block_number = blknr;
  732. delayed_start = blknr;
  733. delayed_extent = blockend;
  734. delayed_buf = buf;
  735. delayed_next = blknr +
  736. (blockend >> log2blksz);
  737. }
  738. } else {
  739. if (previous_block_number != -1) {
  740. /* spill */
  741. put_ext4((uint64_t) ((uint64_t)delayed_start <<
  742. log2blksz),
  743. delayed_buf,
  744. (uint32_t) delayed_extent);
  745. previous_block_number = -1;
  746. }
  747. }
  748. buf += fs->blksz - skipfirst;
  749. }
  750. if (previous_block_number != -1) {
  751. /* spill */
  752. put_ext4((uint64_t) ((uint64_t)delayed_start << log2blksz),
  753. delayed_buf, (uint32_t) delayed_extent);
  754. previous_block_number = -1;
  755. }
  756. return len;
  757. }
  758. int ext4fs_write(const char *fname, const char *buffer,
  759. unsigned long sizebytes, int type)
  760. {
  761. int ret = 0;
  762. struct ext2_inode *file_inode = NULL;
  763. unsigned char *inode_buffer = NULL;
  764. int parent_inodeno;
  765. int inodeno;
  766. time_t timestamp = 0;
  767. uint64_t bytes_reqd_for_file;
  768. unsigned int blks_reqd_for_file;
  769. unsigned int blocks_remaining;
  770. int existing_file_inodeno;
  771. char *temp_ptr = NULL;
  772. long int itable_blkno;
  773. long int parent_itable_blkno;
  774. long int blkoff;
  775. struct ext2_sblock *sblock = &(ext4fs_root->sblock);
  776. unsigned int inodes_per_block;
  777. unsigned int ibmap_idx;
  778. struct ext2_block_group *bgd = NULL;
  779. struct ext_filesystem *fs = get_fs();
  780. ALLOC_CACHE_ALIGN_BUFFER(char, filename, 256);
  781. bool store_link_in_inode = false;
  782. memset(filename, 0x00, 256);
  783. if (type != FILETYPE_REG && type != FILETYPE_SYMLINK)
  784. return -1;
  785. g_parent_inode = zalloc(fs->inodesz);
  786. if (!g_parent_inode)
  787. goto fail;
  788. if (ext4fs_init() != 0) {
  789. printf("error in File System init\n");
  790. return -1;
  791. }
  792. if (le32_to_cpu(fs->sb->feature_ro_compat) & EXT4_FEATURE_RO_COMPAT_METADATA_CSUM) {
  793. printf("Unsupported feature metadata_csum found, not writing.\n");
  794. return -1;
  795. }
  796. inodes_per_block = fs->blksz / fs->inodesz;
  797. parent_inodeno = ext4fs_get_parent_inode_num(fname, filename, F_FILE);
  798. if (parent_inodeno == -1)
  799. goto fail;
  800. if (ext4fs_iget(parent_inodeno, g_parent_inode))
  801. goto fail;
  802. /* do not mess up a directory using hash trees */
  803. if (le32_to_cpu(g_parent_inode->flags) & EXT4_INDEX_FL) {
  804. printf("hash tree directory\n");
  805. goto fail;
  806. }
  807. /* check if the filename is already present in root */
  808. existing_file_inodeno = ext4fs_filename_unlink(filename);
  809. if (existing_file_inodeno != -1) {
  810. ret = ext4fs_delete_file(existing_file_inodeno);
  811. fs->first_pass_bbmap = 0;
  812. fs->curr_blkno = 0;
  813. fs->first_pass_ibmap = 0;
  814. fs->curr_inode_no = 0;
  815. if (ret)
  816. goto fail;
  817. }
  818. /* calculate how many blocks required */
  819. if (type == FILETYPE_SYMLINK &&
  820. sizebytes <= sizeof(file_inode->b.symlink)) {
  821. store_link_in_inode = true;
  822. bytes_reqd_for_file = 0;
  823. } else {
  824. bytes_reqd_for_file = sizebytes;
  825. }
  826. blks_reqd_for_file = lldiv(bytes_reqd_for_file, fs->blksz);
  827. if (do_div(bytes_reqd_for_file, fs->blksz) != 0) {
  828. blks_reqd_for_file++;
  829. debug("total bytes for a file %u\n", blks_reqd_for_file);
  830. }
  831. blocks_remaining = blks_reqd_for_file;
  832. /* test for available space in partition */
  833. if (le32_to_cpu(fs->sb->free_blocks) < blks_reqd_for_file) {
  834. printf("Not enough space on partition !!!\n");
  835. goto fail;
  836. }
  837. inodeno = ext4fs_update_parent_dentry(filename, type);
  838. if (inodeno == -1)
  839. goto fail;
  840. /* prepare file inode */
  841. inode_buffer = zalloc(fs->inodesz);
  842. if (!inode_buffer)
  843. goto fail;
  844. file_inode = (struct ext2_inode *)inode_buffer;
  845. file_inode->size = cpu_to_le32(sizebytes);
  846. if (type == FILETYPE_SYMLINK) {
  847. file_inode->mode = cpu_to_le16(S_IFLNK | S_IRWXU | S_IRWXG |
  848. S_IRWXO);
  849. if (store_link_in_inode) {
  850. strncpy(file_inode->b.symlink, buffer, sizebytes);
  851. sizebytes = 0;
  852. }
  853. } else {
  854. file_inode->mode = cpu_to_le16(S_IFREG | S_IRWXU | S_IRGRP |
  855. S_IROTH | S_IXGRP | S_IXOTH);
  856. }
  857. /* ToDo: Update correct time */
  858. file_inode->mtime = cpu_to_le32(timestamp);
  859. file_inode->atime = cpu_to_le32(timestamp);
  860. file_inode->ctime = cpu_to_le32(timestamp);
  861. file_inode->nlinks = cpu_to_le16(1);
  862. /* Allocate data blocks */
  863. ext4fs_allocate_blocks(file_inode, blocks_remaining,
  864. &blks_reqd_for_file);
  865. file_inode->blockcnt = cpu_to_le32((blks_reqd_for_file * fs->blksz) >>
  866. LOG2_SECTOR_SIZE);
  867. temp_ptr = zalloc(fs->blksz);
  868. if (!temp_ptr)
  869. goto fail;
  870. ibmap_idx = inodeno / le32_to_cpu(ext4fs_root->sblock.inodes_per_group);
  871. inodeno--;
  872. bgd = ext4fs_get_group_descriptor(fs, ibmap_idx);
  873. itable_blkno = ext4fs_bg_get_inode_table_id(bgd, fs) +
  874. (inodeno % le32_to_cpu(sblock->inodes_per_group)) /
  875. inodes_per_block;
  876. blkoff = (inodeno % inodes_per_block) * fs->inodesz;
  877. ext4fs_devread((lbaint_t)itable_blkno * fs->sect_perblk, 0, fs->blksz,
  878. temp_ptr);
  879. if (ext4fs_log_journal(temp_ptr, itable_blkno))
  880. goto fail;
  881. memcpy(temp_ptr + blkoff, inode_buffer, fs->inodesz);
  882. if (ext4fs_put_metadata(temp_ptr, itable_blkno))
  883. goto fail;
  884. /* copy the file content into data blocks */
  885. if (ext4fs_write_file(file_inode, 0, sizebytes, buffer) == -1) {
  886. printf("Error in copying content\n");
  887. /* FIXME: Deallocate data blocks */
  888. goto fail;
  889. }
  890. ibmap_idx = parent_inodeno / le32_to_cpu(ext4fs_root->sblock.inodes_per_group);
  891. parent_inodeno--;
  892. bgd = ext4fs_get_group_descriptor(fs, ibmap_idx);
  893. parent_itable_blkno = ext4fs_bg_get_inode_table_id(bgd, fs) +
  894. (parent_inodeno %
  895. le32_to_cpu(sblock->inodes_per_group)) / inodes_per_block;
  896. blkoff = (parent_inodeno % inodes_per_block) * fs->inodesz;
  897. if (parent_itable_blkno != itable_blkno) {
  898. memset(temp_ptr, '\0', fs->blksz);
  899. ext4fs_devread((lbaint_t)parent_itable_blkno * fs->sect_perblk,
  900. 0, fs->blksz, temp_ptr);
  901. if (ext4fs_log_journal(temp_ptr, parent_itable_blkno))
  902. goto fail;
  903. memcpy(temp_ptr + blkoff, g_parent_inode, fs->inodesz);
  904. if (ext4fs_put_metadata(temp_ptr, parent_itable_blkno))
  905. goto fail;
  906. } else {
  907. /*
  908. * If parent and child fall in same inode table block
  909. * both should be kept in 1 buffer
  910. */
  911. memcpy(temp_ptr + blkoff, g_parent_inode, fs->inodesz);
  912. gd_index--;
  913. if (ext4fs_put_metadata(temp_ptr, itable_blkno))
  914. goto fail;
  915. }
  916. ext4fs_update();
  917. ext4fs_deinit();
  918. fs->first_pass_bbmap = 0;
  919. fs->curr_blkno = 0;
  920. fs->first_pass_ibmap = 0;
  921. fs->curr_inode_no = 0;
  922. free(inode_buffer);
  923. free(g_parent_inode);
  924. free(temp_ptr);
  925. g_parent_inode = NULL;
  926. return 0;
  927. fail:
  928. ext4fs_deinit();
  929. free(inode_buffer);
  930. free(g_parent_inode);
  931. free(temp_ptr);
  932. g_parent_inode = NULL;
  933. return -1;
  934. }
  935. int ext4_write_file(const char *filename, void *buf, loff_t offset,
  936. loff_t len, loff_t *actwrite)
  937. {
  938. int ret;
  939. if (offset != 0) {
  940. printf("** Cannot support non-zero offset **\n");
  941. return -1;
  942. }
  943. ret = ext4fs_write(filename, buf, len, FILETYPE_REG);
  944. if (ret) {
  945. printf("** Error ext4fs_write() **\n");
  946. goto fail;
  947. }
  948. *actwrite = len;
  949. return 0;
  950. fail:
  951. *actwrite = 0;
  952. return -1;
  953. }
  954. int ext4fs_create_link(const char *target, const char *fname)
  955. {
  956. return ext4fs_write(fname, target, strlen(target), FILETYPE_SYMLINK);
  957. }