1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * linux/fs/ufs/balloc.c
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * Copyright (C) 1998
6*4882a593Smuzhiyun * Daniel Pirkl <daniel.pirkl@email.cz>
7*4882a593Smuzhiyun * Charles University, Faculty of Mathematics and Physics
8*4882a593Smuzhiyun *
9*4882a593Smuzhiyun * UFS2 write support Evgeniy Dushistov <dushistov@mail.ru>, 2007
10*4882a593Smuzhiyun */
11*4882a593Smuzhiyun
12*4882a593Smuzhiyun #include <linux/fs.h>
13*4882a593Smuzhiyun #include <linux/stat.h>
14*4882a593Smuzhiyun #include <linux/time.h>
15*4882a593Smuzhiyun #include <linux/string.h>
16*4882a593Smuzhiyun #include <linux/buffer_head.h>
17*4882a593Smuzhiyun #include <linux/capability.h>
18*4882a593Smuzhiyun #include <linux/bitops.h>
19*4882a593Smuzhiyun #include <linux/bio.h>
20*4882a593Smuzhiyun #include <asm/byteorder.h>
21*4882a593Smuzhiyun
22*4882a593Smuzhiyun #include "ufs_fs.h"
23*4882a593Smuzhiyun #include "ufs.h"
24*4882a593Smuzhiyun #include "swab.h"
25*4882a593Smuzhiyun #include "util.h"
26*4882a593Smuzhiyun
27*4882a593Smuzhiyun #define INVBLOCK ((u64)-1L)
28*4882a593Smuzhiyun
29*4882a593Smuzhiyun static u64 ufs_add_fragments(struct inode *, u64, unsigned, unsigned);
30*4882a593Smuzhiyun static u64 ufs_alloc_fragments(struct inode *, unsigned, u64, unsigned, int *);
31*4882a593Smuzhiyun static u64 ufs_alloccg_block(struct inode *, struct ufs_cg_private_info *, u64, int *);
32*4882a593Smuzhiyun static u64 ufs_bitmap_search (struct super_block *, struct ufs_cg_private_info *, u64, unsigned);
33*4882a593Smuzhiyun static unsigned char ufs_fragtable_8fpb[], ufs_fragtable_other[];
34*4882a593Smuzhiyun static void ufs_clusteracct(struct super_block *, struct ufs_cg_private_info *, unsigned, int);
35*4882a593Smuzhiyun
36*4882a593Smuzhiyun /*
37*4882a593Smuzhiyun * Free 'count' fragments from fragment number 'fragment'
38*4882a593Smuzhiyun */
ufs_free_fragments(struct inode * inode,u64 fragment,unsigned count)39*4882a593Smuzhiyun void ufs_free_fragments(struct inode *inode, u64 fragment, unsigned count)
40*4882a593Smuzhiyun {
41*4882a593Smuzhiyun struct super_block * sb;
42*4882a593Smuzhiyun struct ufs_sb_private_info * uspi;
43*4882a593Smuzhiyun struct ufs_cg_private_info * ucpi;
44*4882a593Smuzhiyun struct ufs_cylinder_group * ucg;
45*4882a593Smuzhiyun unsigned cgno, bit, end_bit, bbase, blkmap, i;
46*4882a593Smuzhiyun u64 blkno;
47*4882a593Smuzhiyun
48*4882a593Smuzhiyun sb = inode->i_sb;
49*4882a593Smuzhiyun uspi = UFS_SB(sb)->s_uspi;
50*4882a593Smuzhiyun
51*4882a593Smuzhiyun UFSD("ENTER, fragment %llu, count %u\n",
52*4882a593Smuzhiyun (unsigned long long)fragment, count);
53*4882a593Smuzhiyun
54*4882a593Smuzhiyun if (ufs_fragnum(fragment) + count > uspi->s_fpg)
55*4882a593Smuzhiyun ufs_error (sb, "ufs_free_fragments", "internal error");
56*4882a593Smuzhiyun
57*4882a593Smuzhiyun mutex_lock(&UFS_SB(sb)->s_lock);
58*4882a593Smuzhiyun
59*4882a593Smuzhiyun cgno = ufs_dtog(uspi, fragment);
60*4882a593Smuzhiyun bit = ufs_dtogd(uspi, fragment);
61*4882a593Smuzhiyun if (cgno >= uspi->s_ncg) {
62*4882a593Smuzhiyun ufs_panic (sb, "ufs_free_fragments", "freeing blocks are outside device");
63*4882a593Smuzhiyun goto failed;
64*4882a593Smuzhiyun }
65*4882a593Smuzhiyun
66*4882a593Smuzhiyun ucpi = ufs_load_cylinder (sb, cgno);
67*4882a593Smuzhiyun if (!ucpi)
68*4882a593Smuzhiyun goto failed;
69*4882a593Smuzhiyun ucg = ubh_get_ucg (UCPI_UBH(ucpi));
70*4882a593Smuzhiyun if (!ufs_cg_chkmagic(sb, ucg)) {
71*4882a593Smuzhiyun ufs_panic (sb, "ufs_free_fragments", "internal error, bad magic number on cg %u", cgno);
72*4882a593Smuzhiyun goto failed;
73*4882a593Smuzhiyun }
74*4882a593Smuzhiyun
75*4882a593Smuzhiyun end_bit = bit + count;
76*4882a593Smuzhiyun bbase = ufs_blknum (bit);
77*4882a593Smuzhiyun blkmap = ubh_blkmap (UCPI_UBH(ucpi), ucpi->c_freeoff, bbase);
78*4882a593Smuzhiyun ufs_fragacct (sb, blkmap, ucg->cg_frsum, -1);
79*4882a593Smuzhiyun for (i = bit; i < end_bit; i++) {
80*4882a593Smuzhiyun if (ubh_isclr (UCPI_UBH(ucpi), ucpi->c_freeoff, i))
81*4882a593Smuzhiyun ubh_setbit (UCPI_UBH(ucpi), ucpi->c_freeoff, i);
82*4882a593Smuzhiyun else
83*4882a593Smuzhiyun ufs_error (sb, "ufs_free_fragments",
84*4882a593Smuzhiyun "bit already cleared for fragment %u", i);
85*4882a593Smuzhiyun }
86*4882a593Smuzhiyun
87*4882a593Smuzhiyun inode_sub_bytes(inode, count << uspi->s_fshift);
88*4882a593Smuzhiyun fs32_add(sb, &ucg->cg_cs.cs_nffree, count);
89*4882a593Smuzhiyun uspi->cs_total.cs_nffree += count;
90*4882a593Smuzhiyun fs32_add(sb, &UFS_SB(sb)->fs_cs(cgno).cs_nffree, count);
91*4882a593Smuzhiyun blkmap = ubh_blkmap (UCPI_UBH(ucpi), ucpi->c_freeoff, bbase);
92*4882a593Smuzhiyun ufs_fragacct(sb, blkmap, ucg->cg_frsum, 1);
93*4882a593Smuzhiyun
94*4882a593Smuzhiyun /*
95*4882a593Smuzhiyun * Trying to reassemble free fragments into block
96*4882a593Smuzhiyun */
97*4882a593Smuzhiyun blkno = ufs_fragstoblks (bbase);
98*4882a593Smuzhiyun if (ubh_isblockset(UCPI_UBH(ucpi), ucpi->c_freeoff, blkno)) {
99*4882a593Smuzhiyun fs32_sub(sb, &ucg->cg_cs.cs_nffree, uspi->s_fpb);
100*4882a593Smuzhiyun uspi->cs_total.cs_nffree -= uspi->s_fpb;
101*4882a593Smuzhiyun fs32_sub(sb, &UFS_SB(sb)->fs_cs(cgno).cs_nffree, uspi->s_fpb);
102*4882a593Smuzhiyun if ((UFS_SB(sb)->s_flags & UFS_CG_MASK) == UFS_CG_44BSD)
103*4882a593Smuzhiyun ufs_clusteracct (sb, ucpi, blkno, 1);
104*4882a593Smuzhiyun fs32_add(sb, &ucg->cg_cs.cs_nbfree, 1);
105*4882a593Smuzhiyun uspi->cs_total.cs_nbfree++;
106*4882a593Smuzhiyun fs32_add(sb, &UFS_SB(sb)->fs_cs(cgno).cs_nbfree, 1);
107*4882a593Smuzhiyun if (uspi->fs_magic != UFS2_MAGIC) {
108*4882a593Smuzhiyun unsigned cylno = ufs_cbtocylno (bbase);
109*4882a593Smuzhiyun
110*4882a593Smuzhiyun fs16_add(sb, &ubh_cg_blks(ucpi, cylno,
111*4882a593Smuzhiyun ufs_cbtorpos(bbase)), 1);
112*4882a593Smuzhiyun fs32_add(sb, &ubh_cg_blktot(ucpi, cylno), 1);
113*4882a593Smuzhiyun }
114*4882a593Smuzhiyun }
115*4882a593Smuzhiyun
116*4882a593Smuzhiyun ubh_mark_buffer_dirty (USPI_UBH(uspi));
117*4882a593Smuzhiyun ubh_mark_buffer_dirty (UCPI_UBH(ucpi));
118*4882a593Smuzhiyun if (sb->s_flags & SB_SYNCHRONOUS)
119*4882a593Smuzhiyun ubh_sync_block(UCPI_UBH(ucpi));
120*4882a593Smuzhiyun ufs_mark_sb_dirty(sb);
121*4882a593Smuzhiyun
122*4882a593Smuzhiyun mutex_unlock(&UFS_SB(sb)->s_lock);
123*4882a593Smuzhiyun UFSD("EXIT\n");
124*4882a593Smuzhiyun return;
125*4882a593Smuzhiyun
126*4882a593Smuzhiyun failed:
127*4882a593Smuzhiyun mutex_unlock(&UFS_SB(sb)->s_lock);
128*4882a593Smuzhiyun UFSD("EXIT (FAILED)\n");
129*4882a593Smuzhiyun return;
130*4882a593Smuzhiyun }
131*4882a593Smuzhiyun
132*4882a593Smuzhiyun /*
133*4882a593Smuzhiyun * Free 'count' fragments from fragment number 'fragment' (free whole blocks)
134*4882a593Smuzhiyun */
ufs_free_blocks(struct inode * inode,u64 fragment,unsigned count)135*4882a593Smuzhiyun void ufs_free_blocks(struct inode *inode, u64 fragment, unsigned count)
136*4882a593Smuzhiyun {
137*4882a593Smuzhiyun struct super_block * sb;
138*4882a593Smuzhiyun struct ufs_sb_private_info * uspi;
139*4882a593Smuzhiyun struct ufs_cg_private_info * ucpi;
140*4882a593Smuzhiyun struct ufs_cylinder_group * ucg;
141*4882a593Smuzhiyun unsigned overflow, cgno, bit, end_bit, i;
142*4882a593Smuzhiyun u64 blkno;
143*4882a593Smuzhiyun
144*4882a593Smuzhiyun sb = inode->i_sb;
145*4882a593Smuzhiyun uspi = UFS_SB(sb)->s_uspi;
146*4882a593Smuzhiyun
147*4882a593Smuzhiyun UFSD("ENTER, fragment %llu, count %u\n",
148*4882a593Smuzhiyun (unsigned long long)fragment, count);
149*4882a593Smuzhiyun
150*4882a593Smuzhiyun if ((fragment & uspi->s_fpbmask) || (count & uspi->s_fpbmask)) {
151*4882a593Smuzhiyun ufs_error (sb, "ufs_free_blocks", "internal error, "
152*4882a593Smuzhiyun "fragment %llu, count %u\n",
153*4882a593Smuzhiyun (unsigned long long)fragment, count);
154*4882a593Smuzhiyun goto failed;
155*4882a593Smuzhiyun }
156*4882a593Smuzhiyun
157*4882a593Smuzhiyun mutex_lock(&UFS_SB(sb)->s_lock);
158*4882a593Smuzhiyun
159*4882a593Smuzhiyun do_more:
160*4882a593Smuzhiyun overflow = 0;
161*4882a593Smuzhiyun cgno = ufs_dtog(uspi, fragment);
162*4882a593Smuzhiyun bit = ufs_dtogd(uspi, fragment);
163*4882a593Smuzhiyun if (cgno >= uspi->s_ncg) {
164*4882a593Smuzhiyun ufs_panic (sb, "ufs_free_blocks", "freeing blocks are outside device");
165*4882a593Smuzhiyun goto failed_unlock;
166*4882a593Smuzhiyun }
167*4882a593Smuzhiyun end_bit = bit + count;
168*4882a593Smuzhiyun if (end_bit > uspi->s_fpg) {
169*4882a593Smuzhiyun overflow = bit + count - uspi->s_fpg;
170*4882a593Smuzhiyun count -= overflow;
171*4882a593Smuzhiyun end_bit -= overflow;
172*4882a593Smuzhiyun }
173*4882a593Smuzhiyun
174*4882a593Smuzhiyun ucpi = ufs_load_cylinder (sb, cgno);
175*4882a593Smuzhiyun if (!ucpi)
176*4882a593Smuzhiyun goto failed_unlock;
177*4882a593Smuzhiyun ucg = ubh_get_ucg (UCPI_UBH(ucpi));
178*4882a593Smuzhiyun if (!ufs_cg_chkmagic(sb, ucg)) {
179*4882a593Smuzhiyun ufs_panic (sb, "ufs_free_blocks", "internal error, bad magic number on cg %u", cgno);
180*4882a593Smuzhiyun goto failed_unlock;
181*4882a593Smuzhiyun }
182*4882a593Smuzhiyun
183*4882a593Smuzhiyun for (i = bit; i < end_bit; i += uspi->s_fpb) {
184*4882a593Smuzhiyun blkno = ufs_fragstoblks(i);
185*4882a593Smuzhiyun if (ubh_isblockset(UCPI_UBH(ucpi), ucpi->c_freeoff, blkno)) {
186*4882a593Smuzhiyun ufs_error(sb, "ufs_free_blocks", "freeing free fragment");
187*4882a593Smuzhiyun }
188*4882a593Smuzhiyun ubh_setblock(UCPI_UBH(ucpi), ucpi->c_freeoff, blkno);
189*4882a593Smuzhiyun inode_sub_bytes(inode, uspi->s_fpb << uspi->s_fshift);
190*4882a593Smuzhiyun if ((UFS_SB(sb)->s_flags & UFS_CG_MASK) == UFS_CG_44BSD)
191*4882a593Smuzhiyun ufs_clusteracct (sb, ucpi, blkno, 1);
192*4882a593Smuzhiyun
193*4882a593Smuzhiyun fs32_add(sb, &ucg->cg_cs.cs_nbfree, 1);
194*4882a593Smuzhiyun uspi->cs_total.cs_nbfree++;
195*4882a593Smuzhiyun fs32_add(sb, &UFS_SB(sb)->fs_cs(cgno).cs_nbfree, 1);
196*4882a593Smuzhiyun
197*4882a593Smuzhiyun if (uspi->fs_magic != UFS2_MAGIC) {
198*4882a593Smuzhiyun unsigned cylno = ufs_cbtocylno(i);
199*4882a593Smuzhiyun
200*4882a593Smuzhiyun fs16_add(sb, &ubh_cg_blks(ucpi, cylno,
201*4882a593Smuzhiyun ufs_cbtorpos(i)), 1);
202*4882a593Smuzhiyun fs32_add(sb, &ubh_cg_blktot(ucpi, cylno), 1);
203*4882a593Smuzhiyun }
204*4882a593Smuzhiyun }
205*4882a593Smuzhiyun
206*4882a593Smuzhiyun ubh_mark_buffer_dirty (USPI_UBH(uspi));
207*4882a593Smuzhiyun ubh_mark_buffer_dirty (UCPI_UBH(ucpi));
208*4882a593Smuzhiyun if (sb->s_flags & SB_SYNCHRONOUS)
209*4882a593Smuzhiyun ubh_sync_block(UCPI_UBH(ucpi));
210*4882a593Smuzhiyun
211*4882a593Smuzhiyun if (overflow) {
212*4882a593Smuzhiyun fragment += count;
213*4882a593Smuzhiyun count = overflow;
214*4882a593Smuzhiyun goto do_more;
215*4882a593Smuzhiyun }
216*4882a593Smuzhiyun
217*4882a593Smuzhiyun ufs_mark_sb_dirty(sb);
218*4882a593Smuzhiyun mutex_unlock(&UFS_SB(sb)->s_lock);
219*4882a593Smuzhiyun UFSD("EXIT\n");
220*4882a593Smuzhiyun return;
221*4882a593Smuzhiyun
222*4882a593Smuzhiyun failed_unlock:
223*4882a593Smuzhiyun mutex_unlock(&UFS_SB(sb)->s_lock);
224*4882a593Smuzhiyun failed:
225*4882a593Smuzhiyun UFSD("EXIT (FAILED)\n");
226*4882a593Smuzhiyun return;
227*4882a593Smuzhiyun }
228*4882a593Smuzhiyun
229*4882a593Smuzhiyun /*
230*4882a593Smuzhiyun * Modify inode page cache in such way:
231*4882a593Smuzhiyun * have - blocks with b_blocknr equal to oldb...oldb+count-1
232*4882a593Smuzhiyun * get - blocks with b_blocknr equal to newb...newb+count-1
233*4882a593Smuzhiyun * also we suppose that oldb...oldb+count-1 blocks
234*4882a593Smuzhiyun * situated at the end of file.
235*4882a593Smuzhiyun *
236*4882a593Smuzhiyun * We can come here from ufs_writepage or ufs_prepare_write,
237*4882a593Smuzhiyun * locked_page is argument of these functions, so we already lock it.
238*4882a593Smuzhiyun */
ufs_change_blocknr(struct inode * inode,sector_t beg,unsigned int count,sector_t oldb,sector_t newb,struct page * locked_page)239*4882a593Smuzhiyun static void ufs_change_blocknr(struct inode *inode, sector_t beg,
240*4882a593Smuzhiyun unsigned int count, sector_t oldb,
241*4882a593Smuzhiyun sector_t newb, struct page *locked_page)
242*4882a593Smuzhiyun {
243*4882a593Smuzhiyun const unsigned blks_per_page =
244*4882a593Smuzhiyun 1 << (PAGE_SHIFT - inode->i_blkbits);
245*4882a593Smuzhiyun const unsigned mask = blks_per_page - 1;
246*4882a593Smuzhiyun struct address_space * const mapping = inode->i_mapping;
247*4882a593Smuzhiyun pgoff_t index, cur_index, last_index;
248*4882a593Smuzhiyun unsigned pos, j, lblock;
249*4882a593Smuzhiyun sector_t end, i;
250*4882a593Smuzhiyun struct page *page;
251*4882a593Smuzhiyun struct buffer_head *head, *bh;
252*4882a593Smuzhiyun
253*4882a593Smuzhiyun UFSD("ENTER, ino %lu, count %u, oldb %llu, newb %llu\n",
254*4882a593Smuzhiyun inode->i_ino, count,
255*4882a593Smuzhiyun (unsigned long long)oldb, (unsigned long long)newb);
256*4882a593Smuzhiyun
257*4882a593Smuzhiyun BUG_ON(!locked_page);
258*4882a593Smuzhiyun BUG_ON(!PageLocked(locked_page));
259*4882a593Smuzhiyun
260*4882a593Smuzhiyun cur_index = locked_page->index;
261*4882a593Smuzhiyun end = count + beg;
262*4882a593Smuzhiyun last_index = end >> (PAGE_SHIFT - inode->i_blkbits);
263*4882a593Smuzhiyun for (i = beg; i < end; i = (i | mask) + 1) {
264*4882a593Smuzhiyun index = i >> (PAGE_SHIFT - inode->i_blkbits);
265*4882a593Smuzhiyun
266*4882a593Smuzhiyun if (likely(cur_index != index)) {
267*4882a593Smuzhiyun page = ufs_get_locked_page(mapping, index);
268*4882a593Smuzhiyun if (!page)/* it was truncated */
269*4882a593Smuzhiyun continue;
270*4882a593Smuzhiyun if (IS_ERR(page)) {/* or EIO */
271*4882a593Smuzhiyun ufs_error(inode->i_sb, __func__,
272*4882a593Smuzhiyun "read of page %llu failed\n",
273*4882a593Smuzhiyun (unsigned long long)index);
274*4882a593Smuzhiyun continue;
275*4882a593Smuzhiyun }
276*4882a593Smuzhiyun } else
277*4882a593Smuzhiyun page = locked_page;
278*4882a593Smuzhiyun
279*4882a593Smuzhiyun head = page_buffers(page);
280*4882a593Smuzhiyun bh = head;
281*4882a593Smuzhiyun pos = i & mask;
282*4882a593Smuzhiyun for (j = 0; j < pos; ++j)
283*4882a593Smuzhiyun bh = bh->b_this_page;
284*4882a593Smuzhiyun
285*4882a593Smuzhiyun
286*4882a593Smuzhiyun if (unlikely(index == last_index))
287*4882a593Smuzhiyun lblock = end & mask;
288*4882a593Smuzhiyun else
289*4882a593Smuzhiyun lblock = blks_per_page;
290*4882a593Smuzhiyun
291*4882a593Smuzhiyun do {
292*4882a593Smuzhiyun if (j >= lblock)
293*4882a593Smuzhiyun break;
294*4882a593Smuzhiyun pos = (i - beg) + j;
295*4882a593Smuzhiyun
296*4882a593Smuzhiyun if (!buffer_mapped(bh))
297*4882a593Smuzhiyun map_bh(bh, inode->i_sb, oldb + pos);
298*4882a593Smuzhiyun if (!buffer_uptodate(bh)) {
299*4882a593Smuzhiyun ll_rw_block(REQ_OP_READ, 0, 1, &bh);
300*4882a593Smuzhiyun wait_on_buffer(bh);
301*4882a593Smuzhiyun if (!buffer_uptodate(bh)) {
302*4882a593Smuzhiyun ufs_error(inode->i_sb, __func__,
303*4882a593Smuzhiyun "read of block failed\n");
304*4882a593Smuzhiyun break;
305*4882a593Smuzhiyun }
306*4882a593Smuzhiyun }
307*4882a593Smuzhiyun
308*4882a593Smuzhiyun UFSD(" change from %llu to %llu, pos %u\n",
309*4882a593Smuzhiyun (unsigned long long)(pos + oldb),
310*4882a593Smuzhiyun (unsigned long long)(pos + newb), pos);
311*4882a593Smuzhiyun
312*4882a593Smuzhiyun bh->b_blocknr = newb + pos;
313*4882a593Smuzhiyun clean_bdev_bh_alias(bh);
314*4882a593Smuzhiyun mark_buffer_dirty(bh);
315*4882a593Smuzhiyun ++j;
316*4882a593Smuzhiyun bh = bh->b_this_page;
317*4882a593Smuzhiyun } while (bh != head);
318*4882a593Smuzhiyun
319*4882a593Smuzhiyun if (likely(cur_index != index))
320*4882a593Smuzhiyun ufs_put_locked_page(page);
321*4882a593Smuzhiyun }
322*4882a593Smuzhiyun UFSD("EXIT\n");
323*4882a593Smuzhiyun }
324*4882a593Smuzhiyun
ufs_clear_frags(struct inode * inode,sector_t beg,unsigned int n,int sync)325*4882a593Smuzhiyun static void ufs_clear_frags(struct inode *inode, sector_t beg, unsigned int n,
326*4882a593Smuzhiyun int sync)
327*4882a593Smuzhiyun {
328*4882a593Smuzhiyun struct buffer_head *bh;
329*4882a593Smuzhiyun sector_t end = beg + n;
330*4882a593Smuzhiyun
331*4882a593Smuzhiyun for (; beg < end; ++beg) {
332*4882a593Smuzhiyun bh = sb_getblk(inode->i_sb, beg);
333*4882a593Smuzhiyun lock_buffer(bh);
334*4882a593Smuzhiyun memset(bh->b_data, 0, inode->i_sb->s_blocksize);
335*4882a593Smuzhiyun set_buffer_uptodate(bh);
336*4882a593Smuzhiyun mark_buffer_dirty(bh);
337*4882a593Smuzhiyun unlock_buffer(bh);
338*4882a593Smuzhiyun if (IS_SYNC(inode) || sync)
339*4882a593Smuzhiyun sync_dirty_buffer(bh);
340*4882a593Smuzhiyun brelse(bh);
341*4882a593Smuzhiyun }
342*4882a593Smuzhiyun }
343*4882a593Smuzhiyun
ufs_new_fragments(struct inode * inode,void * p,u64 fragment,u64 goal,unsigned count,int * err,struct page * locked_page)344*4882a593Smuzhiyun u64 ufs_new_fragments(struct inode *inode, void *p, u64 fragment,
345*4882a593Smuzhiyun u64 goal, unsigned count, int *err,
346*4882a593Smuzhiyun struct page *locked_page)
347*4882a593Smuzhiyun {
348*4882a593Smuzhiyun struct super_block * sb;
349*4882a593Smuzhiyun struct ufs_sb_private_info * uspi;
350*4882a593Smuzhiyun struct ufs_super_block_first * usb1;
351*4882a593Smuzhiyun unsigned cgno, oldcount, newcount;
352*4882a593Smuzhiyun u64 tmp, request, result;
353*4882a593Smuzhiyun
354*4882a593Smuzhiyun UFSD("ENTER, ino %lu, fragment %llu, goal %llu, count %u\n",
355*4882a593Smuzhiyun inode->i_ino, (unsigned long long)fragment,
356*4882a593Smuzhiyun (unsigned long long)goal, count);
357*4882a593Smuzhiyun
358*4882a593Smuzhiyun sb = inode->i_sb;
359*4882a593Smuzhiyun uspi = UFS_SB(sb)->s_uspi;
360*4882a593Smuzhiyun usb1 = ubh_get_usb_first(uspi);
361*4882a593Smuzhiyun *err = -ENOSPC;
362*4882a593Smuzhiyun
363*4882a593Smuzhiyun mutex_lock(&UFS_SB(sb)->s_lock);
364*4882a593Smuzhiyun tmp = ufs_data_ptr_to_cpu(sb, p);
365*4882a593Smuzhiyun
366*4882a593Smuzhiyun if (count + ufs_fragnum(fragment) > uspi->s_fpb) {
367*4882a593Smuzhiyun ufs_warning(sb, "ufs_new_fragments", "internal warning"
368*4882a593Smuzhiyun " fragment %llu, count %u",
369*4882a593Smuzhiyun (unsigned long long)fragment, count);
370*4882a593Smuzhiyun count = uspi->s_fpb - ufs_fragnum(fragment);
371*4882a593Smuzhiyun }
372*4882a593Smuzhiyun oldcount = ufs_fragnum (fragment);
373*4882a593Smuzhiyun newcount = oldcount + count;
374*4882a593Smuzhiyun
375*4882a593Smuzhiyun /*
376*4882a593Smuzhiyun * Somebody else has just allocated our fragments
377*4882a593Smuzhiyun */
378*4882a593Smuzhiyun if (oldcount) {
379*4882a593Smuzhiyun if (!tmp) {
380*4882a593Smuzhiyun ufs_error(sb, "ufs_new_fragments", "internal error, "
381*4882a593Smuzhiyun "fragment %llu, tmp %llu\n",
382*4882a593Smuzhiyun (unsigned long long)fragment,
383*4882a593Smuzhiyun (unsigned long long)tmp);
384*4882a593Smuzhiyun mutex_unlock(&UFS_SB(sb)->s_lock);
385*4882a593Smuzhiyun return INVBLOCK;
386*4882a593Smuzhiyun }
387*4882a593Smuzhiyun if (fragment < UFS_I(inode)->i_lastfrag) {
388*4882a593Smuzhiyun UFSD("EXIT (ALREADY ALLOCATED)\n");
389*4882a593Smuzhiyun mutex_unlock(&UFS_SB(sb)->s_lock);
390*4882a593Smuzhiyun return 0;
391*4882a593Smuzhiyun }
392*4882a593Smuzhiyun }
393*4882a593Smuzhiyun else {
394*4882a593Smuzhiyun if (tmp) {
395*4882a593Smuzhiyun UFSD("EXIT (ALREADY ALLOCATED)\n");
396*4882a593Smuzhiyun mutex_unlock(&UFS_SB(sb)->s_lock);
397*4882a593Smuzhiyun return 0;
398*4882a593Smuzhiyun }
399*4882a593Smuzhiyun }
400*4882a593Smuzhiyun
401*4882a593Smuzhiyun /*
402*4882a593Smuzhiyun * There is not enough space for user on the device
403*4882a593Smuzhiyun */
404*4882a593Smuzhiyun if (unlikely(ufs_freefrags(uspi) <= uspi->s_root_blocks)) {
405*4882a593Smuzhiyun if (!capable(CAP_SYS_RESOURCE)) {
406*4882a593Smuzhiyun mutex_unlock(&UFS_SB(sb)->s_lock);
407*4882a593Smuzhiyun UFSD("EXIT (FAILED)\n");
408*4882a593Smuzhiyun return 0;
409*4882a593Smuzhiyun }
410*4882a593Smuzhiyun }
411*4882a593Smuzhiyun
412*4882a593Smuzhiyun if (goal >= uspi->s_size)
413*4882a593Smuzhiyun goal = 0;
414*4882a593Smuzhiyun if (goal == 0)
415*4882a593Smuzhiyun cgno = ufs_inotocg (inode->i_ino);
416*4882a593Smuzhiyun else
417*4882a593Smuzhiyun cgno = ufs_dtog(uspi, goal);
418*4882a593Smuzhiyun
419*4882a593Smuzhiyun /*
420*4882a593Smuzhiyun * allocate new fragment
421*4882a593Smuzhiyun */
422*4882a593Smuzhiyun if (oldcount == 0) {
423*4882a593Smuzhiyun result = ufs_alloc_fragments (inode, cgno, goal, count, err);
424*4882a593Smuzhiyun if (result) {
425*4882a593Smuzhiyun ufs_clear_frags(inode, result + oldcount,
426*4882a593Smuzhiyun newcount - oldcount, locked_page != NULL);
427*4882a593Smuzhiyun *err = 0;
428*4882a593Smuzhiyun write_seqlock(&UFS_I(inode)->meta_lock);
429*4882a593Smuzhiyun ufs_cpu_to_data_ptr(sb, p, result);
430*4882a593Smuzhiyun UFS_I(inode)->i_lastfrag =
431*4882a593Smuzhiyun max(UFS_I(inode)->i_lastfrag, fragment + count);
432*4882a593Smuzhiyun write_sequnlock(&UFS_I(inode)->meta_lock);
433*4882a593Smuzhiyun }
434*4882a593Smuzhiyun mutex_unlock(&UFS_SB(sb)->s_lock);
435*4882a593Smuzhiyun UFSD("EXIT, result %llu\n", (unsigned long long)result);
436*4882a593Smuzhiyun return result;
437*4882a593Smuzhiyun }
438*4882a593Smuzhiyun
439*4882a593Smuzhiyun /*
440*4882a593Smuzhiyun * resize block
441*4882a593Smuzhiyun */
442*4882a593Smuzhiyun result = ufs_add_fragments(inode, tmp, oldcount, newcount);
443*4882a593Smuzhiyun if (result) {
444*4882a593Smuzhiyun *err = 0;
445*4882a593Smuzhiyun read_seqlock_excl(&UFS_I(inode)->meta_lock);
446*4882a593Smuzhiyun UFS_I(inode)->i_lastfrag = max(UFS_I(inode)->i_lastfrag,
447*4882a593Smuzhiyun fragment + count);
448*4882a593Smuzhiyun read_sequnlock_excl(&UFS_I(inode)->meta_lock);
449*4882a593Smuzhiyun ufs_clear_frags(inode, result + oldcount, newcount - oldcount,
450*4882a593Smuzhiyun locked_page != NULL);
451*4882a593Smuzhiyun mutex_unlock(&UFS_SB(sb)->s_lock);
452*4882a593Smuzhiyun UFSD("EXIT, result %llu\n", (unsigned long long)result);
453*4882a593Smuzhiyun return result;
454*4882a593Smuzhiyun }
455*4882a593Smuzhiyun
456*4882a593Smuzhiyun /*
457*4882a593Smuzhiyun * allocate new block and move data
458*4882a593Smuzhiyun */
459*4882a593Smuzhiyun if (fs32_to_cpu(sb, usb1->fs_optim) == UFS_OPTSPACE) {
460*4882a593Smuzhiyun request = newcount;
461*4882a593Smuzhiyun if (uspi->cs_total.cs_nffree < uspi->s_space_to_time)
462*4882a593Smuzhiyun usb1->fs_optim = cpu_to_fs32(sb, UFS_OPTTIME);
463*4882a593Smuzhiyun } else {
464*4882a593Smuzhiyun request = uspi->s_fpb;
465*4882a593Smuzhiyun if (uspi->cs_total.cs_nffree > uspi->s_time_to_space)
466*4882a593Smuzhiyun usb1->fs_optim = cpu_to_fs32(sb, UFS_OPTSPACE);
467*4882a593Smuzhiyun }
468*4882a593Smuzhiyun result = ufs_alloc_fragments (inode, cgno, goal, request, err);
469*4882a593Smuzhiyun if (result) {
470*4882a593Smuzhiyun ufs_clear_frags(inode, result + oldcount, newcount - oldcount,
471*4882a593Smuzhiyun locked_page != NULL);
472*4882a593Smuzhiyun mutex_unlock(&UFS_SB(sb)->s_lock);
473*4882a593Smuzhiyun ufs_change_blocknr(inode, fragment - oldcount, oldcount,
474*4882a593Smuzhiyun uspi->s_sbbase + tmp,
475*4882a593Smuzhiyun uspi->s_sbbase + result, locked_page);
476*4882a593Smuzhiyun *err = 0;
477*4882a593Smuzhiyun write_seqlock(&UFS_I(inode)->meta_lock);
478*4882a593Smuzhiyun ufs_cpu_to_data_ptr(sb, p, result);
479*4882a593Smuzhiyun UFS_I(inode)->i_lastfrag = max(UFS_I(inode)->i_lastfrag,
480*4882a593Smuzhiyun fragment + count);
481*4882a593Smuzhiyun write_sequnlock(&UFS_I(inode)->meta_lock);
482*4882a593Smuzhiyun if (newcount < request)
483*4882a593Smuzhiyun ufs_free_fragments (inode, result + newcount, request - newcount);
484*4882a593Smuzhiyun ufs_free_fragments (inode, tmp, oldcount);
485*4882a593Smuzhiyun UFSD("EXIT, result %llu\n", (unsigned long long)result);
486*4882a593Smuzhiyun return result;
487*4882a593Smuzhiyun }
488*4882a593Smuzhiyun
489*4882a593Smuzhiyun mutex_unlock(&UFS_SB(sb)->s_lock);
490*4882a593Smuzhiyun UFSD("EXIT (FAILED)\n");
491*4882a593Smuzhiyun return 0;
492*4882a593Smuzhiyun }
493*4882a593Smuzhiyun
try_add_frags(struct inode * inode,unsigned frags)494*4882a593Smuzhiyun static bool try_add_frags(struct inode *inode, unsigned frags)
495*4882a593Smuzhiyun {
496*4882a593Smuzhiyun unsigned size = frags * i_blocksize(inode);
497*4882a593Smuzhiyun spin_lock(&inode->i_lock);
498*4882a593Smuzhiyun __inode_add_bytes(inode, size);
499*4882a593Smuzhiyun if (unlikely((u32)inode->i_blocks != inode->i_blocks)) {
500*4882a593Smuzhiyun __inode_sub_bytes(inode, size);
501*4882a593Smuzhiyun spin_unlock(&inode->i_lock);
502*4882a593Smuzhiyun return false;
503*4882a593Smuzhiyun }
504*4882a593Smuzhiyun spin_unlock(&inode->i_lock);
505*4882a593Smuzhiyun return true;
506*4882a593Smuzhiyun }
507*4882a593Smuzhiyun
ufs_add_fragments(struct inode * inode,u64 fragment,unsigned oldcount,unsigned newcount)508*4882a593Smuzhiyun static u64 ufs_add_fragments(struct inode *inode, u64 fragment,
509*4882a593Smuzhiyun unsigned oldcount, unsigned newcount)
510*4882a593Smuzhiyun {
511*4882a593Smuzhiyun struct super_block * sb;
512*4882a593Smuzhiyun struct ufs_sb_private_info * uspi;
513*4882a593Smuzhiyun struct ufs_cg_private_info * ucpi;
514*4882a593Smuzhiyun struct ufs_cylinder_group * ucg;
515*4882a593Smuzhiyun unsigned cgno, fragno, fragoff, count, fragsize, i;
516*4882a593Smuzhiyun
517*4882a593Smuzhiyun UFSD("ENTER, fragment %llu, oldcount %u, newcount %u\n",
518*4882a593Smuzhiyun (unsigned long long)fragment, oldcount, newcount);
519*4882a593Smuzhiyun
520*4882a593Smuzhiyun sb = inode->i_sb;
521*4882a593Smuzhiyun uspi = UFS_SB(sb)->s_uspi;
522*4882a593Smuzhiyun count = newcount - oldcount;
523*4882a593Smuzhiyun
524*4882a593Smuzhiyun cgno = ufs_dtog(uspi, fragment);
525*4882a593Smuzhiyun if (fs32_to_cpu(sb, UFS_SB(sb)->fs_cs(cgno).cs_nffree) < count)
526*4882a593Smuzhiyun return 0;
527*4882a593Smuzhiyun if ((ufs_fragnum (fragment) + newcount) > uspi->s_fpb)
528*4882a593Smuzhiyun return 0;
529*4882a593Smuzhiyun ucpi = ufs_load_cylinder (sb, cgno);
530*4882a593Smuzhiyun if (!ucpi)
531*4882a593Smuzhiyun return 0;
532*4882a593Smuzhiyun ucg = ubh_get_ucg (UCPI_UBH(ucpi));
533*4882a593Smuzhiyun if (!ufs_cg_chkmagic(sb, ucg)) {
534*4882a593Smuzhiyun ufs_panic (sb, "ufs_add_fragments",
535*4882a593Smuzhiyun "internal error, bad magic number on cg %u", cgno);
536*4882a593Smuzhiyun return 0;
537*4882a593Smuzhiyun }
538*4882a593Smuzhiyun
539*4882a593Smuzhiyun fragno = ufs_dtogd(uspi, fragment);
540*4882a593Smuzhiyun fragoff = ufs_fragnum (fragno);
541*4882a593Smuzhiyun for (i = oldcount; i < newcount; i++)
542*4882a593Smuzhiyun if (ubh_isclr (UCPI_UBH(ucpi), ucpi->c_freeoff, fragno + i))
543*4882a593Smuzhiyun return 0;
544*4882a593Smuzhiyun
545*4882a593Smuzhiyun if (!try_add_frags(inode, count))
546*4882a593Smuzhiyun return 0;
547*4882a593Smuzhiyun /*
548*4882a593Smuzhiyun * Block can be extended
549*4882a593Smuzhiyun */
550*4882a593Smuzhiyun ucg->cg_time = ufs_get_seconds(sb);
551*4882a593Smuzhiyun for (i = newcount; i < (uspi->s_fpb - fragoff); i++)
552*4882a593Smuzhiyun if (ubh_isclr (UCPI_UBH(ucpi), ucpi->c_freeoff, fragno + i))
553*4882a593Smuzhiyun break;
554*4882a593Smuzhiyun fragsize = i - oldcount;
555*4882a593Smuzhiyun if (!fs32_to_cpu(sb, ucg->cg_frsum[fragsize]))
556*4882a593Smuzhiyun ufs_panic (sb, "ufs_add_fragments",
557*4882a593Smuzhiyun "internal error or corrupted bitmap on cg %u", cgno);
558*4882a593Smuzhiyun fs32_sub(sb, &ucg->cg_frsum[fragsize], 1);
559*4882a593Smuzhiyun if (fragsize != count)
560*4882a593Smuzhiyun fs32_add(sb, &ucg->cg_frsum[fragsize - count], 1);
561*4882a593Smuzhiyun for (i = oldcount; i < newcount; i++)
562*4882a593Smuzhiyun ubh_clrbit (UCPI_UBH(ucpi), ucpi->c_freeoff, fragno + i);
563*4882a593Smuzhiyun
564*4882a593Smuzhiyun fs32_sub(sb, &ucg->cg_cs.cs_nffree, count);
565*4882a593Smuzhiyun fs32_sub(sb, &UFS_SB(sb)->fs_cs(cgno).cs_nffree, count);
566*4882a593Smuzhiyun uspi->cs_total.cs_nffree -= count;
567*4882a593Smuzhiyun
568*4882a593Smuzhiyun ubh_mark_buffer_dirty (USPI_UBH(uspi));
569*4882a593Smuzhiyun ubh_mark_buffer_dirty (UCPI_UBH(ucpi));
570*4882a593Smuzhiyun if (sb->s_flags & SB_SYNCHRONOUS)
571*4882a593Smuzhiyun ubh_sync_block(UCPI_UBH(ucpi));
572*4882a593Smuzhiyun ufs_mark_sb_dirty(sb);
573*4882a593Smuzhiyun
574*4882a593Smuzhiyun UFSD("EXIT, fragment %llu\n", (unsigned long long)fragment);
575*4882a593Smuzhiyun
576*4882a593Smuzhiyun return fragment;
577*4882a593Smuzhiyun }
578*4882a593Smuzhiyun
579*4882a593Smuzhiyun #define UFS_TEST_FREE_SPACE_CG \
580*4882a593Smuzhiyun ucg = (struct ufs_cylinder_group *) UFS_SB(sb)->s_ucg[cgno]->b_data; \
581*4882a593Smuzhiyun if (fs32_to_cpu(sb, ucg->cg_cs.cs_nbfree)) \
582*4882a593Smuzhiyun goto cg_found; \
583*4882a593Smuzhiyun for (k = count; k < uspi->s_fpb; k++) \
584*4882a593Smuzhiyun if (fs32_to_cpu(sb, ucg->cg_frsum[k])) \
585*4882a593Smuzhiyun goto cg_found;
586*4882a593Smuzhiyun
ufs_alloc_fragments(struct inode * inode,unsigned cgno,u64 goal,unsigned count,int * err)587*4882a593Smuzhiyun static u64 ufs_alloc_fragments(struct inode *inode, unsigned cgno,
588*4882a593Smuzhiyun u64 goal, unsigned count, int *err)
589*4882a593Smuzhiyun {
590*4882a593Smuzhiyun struct super_block * sb;
591*4882a593Smuzhiyun struct ufs_sb_private_info * uspi;
592*4882a593Smuzhiyun struct ufs_cg_private_info * ucpi;
593*4882a593Smuzhiyun struct ufs_cylinder_group * ucg;
594*4882a593Smuzhiyun unsigned oldcg, i, j, k, allocsize;
595*4882a593Smuzhiyun u64 result;
596*4882a593Smuzhiyun
597*4882a593Smuzhiyun UFSD("ENTER, ino %lu, cgno %u, goal %llu, count %u\n",
598*4882a593Smuzhiyun inode->i_ino, cgno, (unsigned long long)goal, count);
599*4882a593Smuzhiyun
600*4882a593Smuzhiyun sb = inode->i_sb;
601*4882a593Smuzhiyun uspi = UFS_SB(sb)->s_uspi;
602*4882a593Smuzhiyun oldcg = cgno;
603*4882a593Smuzhiyun
604*4882a593Smuzhiyun /*
605*4882a593Smuzhiyun * 1. searching on preferred cylinder group
606*4882a593Smuzhiyun */
607*4882a593Smuzhiyun UFS_TEST_FREE_SPACE_CG
608*4882a593Smuzhiyun
609*4882a593Smuzhiyun /*
610*4882a593Smuzhiyun * 2. quadratic rehash
611*4882a593Smuzhiyun */
612*4882a593Smuzhiyun for (j = 1; j < uspi->s_ncg; j *= 2) {
613*4882a593Smuzhiyun cgno += j;
614*4882a593Smuzhiyun if (cgno >= uspi->s_ncg)
615*4882a593Smuzhiyun cgno -= uspi->s_ncg;
616*4882a593Smuzhiyun UFS_TEST_FREE_SPACE_CG
617*4882a593Smuzhiyun }
618*4882a593Smuzhiyun
619*4882a593Smuzhiyun /*
620*4882a593Smuzhiyun * 3. brute force search
621*4882a593Smuzhiyun * We start at i = 2 ( 0 is checked at 1.step, 1 at 2.step )
622*4882a593Smuzhiyun */
623*4882a593Smuzhiyun cgno = (oldcg + 1) % uspi->s_ncg;
624*4882a593Smuzhiyun for (j = 2; j < uspi->s_ncg; j++) {
625*4882a593Smuzhiyun cgno++;
626*4882a593Smuzhiyun if (cgno >= uspi->s_ncg)
627*4882a593Smuzhiyun cgno = 0;
628*4882a593Smuzhiyun UFS_TEST_FREE_SPACE_CG
629*4882a593Smuzhiyun }
630*4882a593Smuzhiyun
631*4882a593Smuzhiyun UFSD("EXIT (FAILED)\n");
632*4882a593Smuzhiyun return 0;
633*4882a593Smuzhiyun
634*4882a593Smuzhiyun cg_found:
635*4882a593Smuzhiyun ucpi = ufs_load_cylinder (sb, cgno);
636*4882a593Smuzhiyun if (!ucpi)
637*4882a593Smuzhiyun return 0;
638*4882a593Smuzhiyun ucg = ubh_get_ucg (UCPI_UBH(ucpi));
639*4882a593Smuzhiyun if (!ufs_cg_chkmagic(sb, ucg))
640*4882a593Smuzhiyun ufs_panic (sb, "ufs_alloc_fragments",
641*4882a593Smuzhiyun "internal error, bad magic number on cg %u", cgno);
642*4882a593Smuzhiyun ucg->cg_time = ufs_get_seconds(sb);
643*4882a593Smuzhiyun
644*4882a593Smuzhiyun if (count == uspi->s_fpb) {
645*4882a593Smuzhiyun result = ufs_alloccg_block (inode, ucpi, goal, err);
646*4882a593Smuzhiyun if (result == INVBLOCK)
647*4882a593Smuzhiyun return 0;
648*4882a593Smuzhiyun goto succed;
649*4882a593Smuzhiyun }
650*4882a593Smuzhiyun
651*4882a593Smuzhiyun for (allocsize = count; allocsize < uspi->s_fpb; allocsize++)
652*4882a593Smuzhiyun if (fs32_to_cpu(sb, ucg->cg_frsum[allocsize]) != 0)
653*4882a593Smuzhiyun break;
654*4882a593Smuzhiyun
655*4882a593Smuzhiyun if (allocsize == uspi->s_fpb) {
656*4882a593Smuzhiyun result = ufs_alloccg_block (inode, ucpi, goal, err);
657*4882a593Smuzhiyun if (result == INVBLOCK)
658*4882a593Smuzhiyun return 0;
659*4882a593Smuzhiyun goal = ufs_dtogd(uspi, result);
660*4882a593Smuzhiyun for (i = count; i < uspi->s_fpb; i++)
661*4882a593Smuzhiyun ubh_setbit (UCPI_UBH(ucpi), ucpi->c_freeoff, goal + i);
662*4882a593Smuzhiyun i = uspi->s_fpb - count;
663*4882a593Smuzhiyun
664*4882a593Smuzhiyun inode_sub_bytes(inode, i << uspi->s_fshift);
665*4882a593Smuzhiyun fs32_add(sb, &ucg->cg_cs.cs_nffree, i);
666*4882a593Smuzhiyun uspi->cs_total.cs_nffree += i;
667*4882a593Smuzhiyun fs32_add(sb, &UFS_SB(sb)->fs_cs(cgno).cs_nffree, i);
668*4882a593Smuzhiyun fs32_add(sb, &ucg->cg_frsum[i], 1);
669*4882a593Smuzhiyun goto succed;
670*4882a593Smuzhiyun }
671*4882a593Smuzhiyun
672*4882a593Smuzhiyun result = ufs_bitmap_search (sb, ucpi, goal, allocsize);
673*4882a593Smuzhiyun if (result == INVBLOCK)
674*4882a593Smuzhiyun return 0;
675*4882a593Smuzhiyun if (!try_add_frags(inode, count))
676*4882a593Smuzhiyun return 0;
677*4882a593Smuzhiyun for (i = 0; i < count; i++)
678*4882a593Smuzhiyun ubh_clrbit (UCPI_UBH(ucpi), ucpi->c_freeoff, result + i);
679*4882a593Smuzhiyun
680*4882a593Smuzhiyun fs32_sub(sb, &ucg->cg_cs.cs_nffree, count);
681*4882a593Smuzhiyun uspi->cs_total.cs_nffree -= count;
682*4882a593Smuzhiyun fs32_sub(sb, &UFS_SB(sb)->fs_cs(cgno).cs_nffree, count);
683*4882a593Smuzhiyun fs32_sub(sb, &ucg->cg_frsum[allocsize], 1);
684*4882a593Smuzhiyun
685*4882a593Smuzhiyun if (count != allocsize)
686*4882a593Smuzhiyun fs32_add(sb, &ucg->cg_frsum[allocsize - count], 1);
687*4882a593Smuzhiyun
688*4882a593Smuzhiyun succed:
689*4882a593Smuzhiyun ubh_mark_buffer_dirty (USPI_UBH(uspi));
690*4882a593Smuzhiyun ubh_mark_buffer_dirty (UCPI_UBH(ucpi));
691*4882a593Smuzhiyun if (sb->s_flags & SB_SYNCHRONOUS)
692*4882a593Smuzhiyun ubh_sync_block(UCPI_UBH(ucpi));
693*4882a593Smuzhiyun ufs_mark_sb_dirty(sb);
694*4882a593Smuzhiyun
695*4882a593Smuzhiyun result += cgno * uspi->s_fpg;
696*4882a593Smuzhiyun UFSD("EXIT3, result %llu\n", (unsigned long long)result);
697*4882a593Smuzhiyun return result;
698*4882a593Smuzhiyun }
699*4882a593Smuzhiyun
ufs_alloccg_block(struct inode * inode,struct ufs_cg_private_info * ucpi,u64 goal,int * err)700*4882a593Smuzhiyun static u64 ufs_alloccg_block(struct inode *inode,
701*4882a593Smuzhiyun struct ufs_cg_private_info *ucpi,
702*4882a593Smuzhiyun u64 goal, int *err)
703*4882a593Smuzhiyun {
704*4882a593Smuzhiyun struct super_block * sb;
705*4882a593Smuzhiyun struct ufs_sb_private_info * uspi;
706*4882a593Smuzhiyun struct ufs_cylinder_group * ucg;
707*4882a593Smuzhiyun u64 result, blkno;
708*4882a593Smuzhiyun
709*4882a593Smuzhiyun UFSD("ENTER, goal %llu\n", (unsigned long long)goal);
710*4882a593Smuzhiyun
711*4882a593Smuzhiyun sb = inode->i_sb;
712*4882a593Smuzhiyun uspi = UFS_SB(sb)->s_uspi;
713*4882a593Smuzhiyun ucg = ubh_get_ucg(UCPI_UBH(ucpi));
714*4882a593Smuzhiyun
715*4882a593Smuzhiyun if (goal == 0) {
716*4882a593Smuzhiyun goal = ucpi->c_rotor;
717*4882a593Smuzhiyun goto norot;
718*4882a593Smuzhiyun }
719*4882a593Smuzhiyun goal = ufs_blknum (goal);
720*4882a593Smuzhiyun goal = ufs_dtogd(uspi, goal);
721*4882a593Smuzhiyun
722*4882a593Smuzhiyun /*
723*4882a593Smuzhiyun * If the requested block is available, use it.
724*4882a593Smuzhiyun */
725*4882a593Smuzhiyun if (ubh_isblockset(UCPI_UBH(ucpi), ucpi->c_freeoff, ufs_fragstoblks(goal))) {
726*4882a593Smuzhiyun result = goal;
727*4882a593Smuzhiyun goto gotit;
728*4882a593Smuzhiyun }
729*4882a593Smuzhiyun
730*4882a593Smuzhiyun norot:
731*4882a593Smuzhiyun result = ufs_bitmap_search (sb, ucpi, goal, uspi->s_fpb);
732*4882a593Smuzhiyun if (result == INVBLOCK)
733*4882a593Smuzhiyun return INVBLOCK;
734*4882a593Smuzhiyun ucpi->c_rotor = result;
735*4882a593Smuzhiyun gotit:
736*4882a593Smuzhiyun if (!try_add_frags(inode, uspi->s_fpb))
737*4882a593Smuzhiyun return 0;
738*4882a593Smuzhiyun blkno = ufs_fragstoblks(result);
739*4882a593Smuzhiyun ubh_clrblock (UCPI_UBH(ucpi), ucpi->c_freeoff, blkno);
740*4882a593Smuzhiyun if ((UFS_SB(sb)->s_flags & UFS_CG_MASK) == UFS_CG_44BSD)
741*4882a593Smuzhiyun ufs_clusteracct (sb, ucpi, blkno, -1);
742*4882a593Smuzhiyun
743*4882a593Smuzhiyun fs32_sub(sb, &ucg->cg_cs.cs_nbfree, 1);
744*4882a593Smuzhiyun uspi->cs_total.cs_nbfree--;
745*4882a593Smuzhiyun fs32_sub(sb, &UFS_SB(sb)->fs_cs(ucpi->c_cgx).cs_nbfree, 1);
746*4882a593Smuzhiyun
747*4882a593Smuzhiyun if (uspi->fs_magic != UFS2_MAGIC) {
748*4882a593Smuzhiyun unsigned cylno = ufs_cbtocylno((unsigned)result);
749*4882a593Smuzhiyun
750*4882a593Smuzhiyun fs16_sub(sb, &ubh_cg_blks(ucpi, cylno,
751*4882a593Smuzhiyun ufs_cbtorpos((unsigned)result)), 1);
752*4882a593Smuzhiyun fs32_sub(sb, &ubh_cg_blktot(ucpi, cylno), 1);
753*4882a593Smuzhiyun }
754*4882a593Smuzhiyun
755*4882a593Smuzhiyun UFSD("EXIT, result %llu\n", (unsigned long long)result);
756*4882a593Smuzhiyun
757*4882a593Smuzhiyun return result;
758*4882a593Smuzhiyun }
759*4882a593Smuzhiyun
ubh_scanc(struct ufs_sb_private_info * uspi,struct ufs_buffer_head * ubh,unsigned begin,unsigned size,unsigned char * table,unsigned char mask)760*4882a593Smuzhiyun static unsigned ubh_scanc(struct ufs_sb_private_info *uspi,
761*4882a593Smuzhiyun struct ufs_buffer_head *ubh,
762*4882a593Smuzhiyun unsigned begin, unsigned size,
763*4882a593Smuzhiyun unsigned char *table, unsigned char mask)
764*4882a593Smuzhiyun {
765*4882a593Smuzhiyun unsigned rest, offset;
766*4882a593Smuzhiyun unsigned char *cp;
767*4882a593Smuzhiyun
768*4882a593Smuzhiyun
769*4882a593Smuzhiyun offset = begin & ~uspi->s_fmask;
770*4882a593Smuzhiyun begin >>= uspi->s_fshift;
771*4882a593Smuzhiyun for (;;) {
772*4882a593Smuzhiyun if ((offset + size) < uspi->s_fsize)
773*4882a593Smuzhiyun rest = size;
774*4882a593Smuzhiyun else
775*4882a593Smuzhiyun rest = uspi->s_fsize - offset;
776*4882a593Smuzhiyun size -= rest;
777*4882a593Smuzhiyun cp = ubh->bh[begin]->b_data + offset;
778*4882a593Smuzhiyun while ((table[*cp++] & mask) == 0 && --rest)
779*4882a593Smuzhiyun ;
780*4882a593Smuzhiyun if (rest || !size)
781*4882a593Smuzhiyun break;
782*4882a593Smuzhiyun begin++;
783*4882a593Smuzhiyun offset = 0;
784*4882a593Smuzhiyun }
785*4882a593Smuzhiyun return (size + rest);
786*4882a593Smuzhiyun }
787*4882a593Smuzhiyun
788*4882a593Smuzhiyun /*
789*4882a593Smuzhiyun * Find a block of the specified size in the specified cylinder group.
790*4882a593Smuzhiyun * @sp: pointer to super block
791*4882a593Smuzhiyun * @ucpi: pointer to cylinder group info
792*4882a593Smuzhiyun * @goal: near which block we want find new one
793*4882a593Smuzhiyun * @count: specified size
794*4882a593Smuzhiyun */
ufs_bitmap_search(struct super_block * sb,struct ufs_cg_private_info * ucpi,u64 goal,unsigned count)795*4882a593Smuzhiyun static u64 ufs_bitmap_search(struct super_block *sb,
796*4882a593Smuzhiyun struct ufs_cg_private_info *ucpi,
797*4882a593Smuzhiyun u64 goal, unsigned count)
798*4882a593Smuzhiyun {
799*4882a593Smuzhiyun /*
800*4882a593Smuzhiyun * Bit patterns for identifying fragments in the block map
801*4882a593Smuzhiyun * used as ((map & mask_arr) == want_arr)
802*4882a593Smuzhiyun */
803*4882a593Smuzhiyun static const int mask_arr[9] = {
804*4882a593Smuzhiyun 0x3, 0x7, 0xf, 0x1f, 0x3f, 0x7f, 0xff, 0x1ff, 0x3ff
805*4882a593Smuzhiyun };
806*4882a593Smuzhiyun static const int want_arr[9] = {
807*4882a593Smuzhiyun 0x0, 0x2, 0x6, 0xe, 0x1e, 0x3e, 0x7e, 0xfe, 0x1fe
808*4882a593Smuzhiyun };
809*4882a593Smuzhiyun struct ufs_sb_private_info *uspi = UFS_SB(sb)->s_uspi;
810*4882a593Smuzhiyun unsigned start, length, loc;
811*4882a593Smuzhiyun unsigned pos, want, blockmap, mask, end;
812*4882a593Smuzhiyun u64 result;
813*4882a593Smuzhiyun
814*4882a593Smuzhiyun UFSD("ENTER, cg %u, goal %llu, count %u\n", ucpi->c_cgx,
815*4882a593Smuzhiyun (unsigned long long)goal, count);
816*4882a593Smuzhiyun
817*4882a593Smuzhiyun if (goal)
818*4882a593Smuzhiyun start = ufs_dtogd(uspi, goal) >> 3;
819*4882a593Smuzhiyun else
820*4882a593Smuzhiyun start = ucpi->c_frotor >> 3;
821*4882a593Smuzhiyun
822*4882a593Smuzhiyun length = ((uspi->s_fpg + 7) >> 3) - start;
823*4882a593Smuzhiyun loc = ubh_scanc(uspi, UCPI_UBH(ucpi), ucpi->c_freeoff + start, length,
824*4882a593Smuzhiyun (uspi->s_fpb == 8) ? ufs_fragtable_8fpb : ufs_fragtable_other,
825*4882a593Smuzhiyun 1 << (count - 1 + (uspi->s_fpb & 7)));
826*4882a593Smuzhiyun if (loc == 0) {
827*4882a593Smuzhiyun length = start + 1;
828*4882a593Smuzhiyun loc = ubh_scanc(uspi, UCPI_UBH(ucpi), ucpi->c_freeoff, length,
829*4882a593Smuzhiyun (uspi->s_fpb == 8) ? ufs_fragtable_8fpb :
830*4882a593Smuzhiyun ufs_fragtable_other,
831*4882a593Smuzhiyun 1 << (count - 1 + (uspi->s_fpb & 7)));
832*4882a593Smuzhiyun if (loc == 0) {
833*4882a593Smuzhiyun ufs_error(sb, "ufs_bitmap_search",
834*4882a593Smuzhiyun "bitmap corrupted on cg %u, start %u,"
835*4882a593Smuzhiyun " length %u, count %u, freeoff %u\n",
836*4882a593Smuzhiyun ucpi->c_cgx, start, length, count,
837*4882a593Smuzhiyun ucpi->c_freeoff);
838*4882a593Smuzhiyun return INVBLOCK;
839*4882a593Smuzhiyun }
840*4882a593Smuzhiyun start = 0;
841*4882a593Smuzhiyun }
842*4882a593Smuzhiyun result = (start + length - loc) << 3;
843*4882a593Smuzhiyun ucpi->c_frotor = result;
844*4882a593Smuzhiyun
845*4882a593Smuzhiyun /*
846*4882a593Smuzhiyun * found the byte in the map
847*4882a593Smuzhiyun */
848*4882a593Smuzhiyun
849*4882a593Smuzhiyun for (end = result + 8; result < end; result += uspi->s_fpb) {
850*4882a593Smuzhiyun blockmap = ubh_blkmap(UCPI_UBH(ucpi), ucpi->c_freeoff, result);
851*4882a593Smuzhiyun blockmap <<= 1;
852*4882a593Smuzhiyun mask = mask_arr[count];
853*4882a593Smuzhiyun want = want_arr[count];
854*4882a593Smuzhiyun for (pos = 0; pos <= uspi->s_fpb - count; pos++) {
855*4882a593Smuzhiyun if ((blockmap & mask) == want) {
856*4882a593Smuzhiyun UFSD("EXIT, result %llu\n",
857*4882a593Smuzhiyun (unsigned long long)result);
858*4882a593Smuzhiyun return result + pos;
859*4882a593Smuzhiyun }
860*4882a593Smuzhiyun mask <<= 1;
861*4882a593Smuzhiyun want <<= 1;
862*4882a593Smuzhiyun }
863*4882a593Smuzhiyun }
864*4882a593Smuzhiyun
865*4882a593Smuzhiyun ufs_error(sb, "ufs_bitmap_search", "block not in map on cg %u\n",
866*4882a593Smuzhiyun ucpi->c_cgx);
867*4882a593Smuzhiyun UFSD("EXIT (FAILED)\n");
868*4882a593Smuzhiyun return INVBLOCK;
869*4882a593Smuzhiyun }
870*4882a593Smuzhiyun
ufs_clusteracct(struct super_block * sb,struct ufs_cg_private_info * ucpi,unsigned blkno,int cnt)871*4882a593Smuzhiyun static void ufs_clusteracct(struct super_block * sb,
872*4882a593Smuzhiyun struct ufs_cg_private_info * ucpi, unsigned blkno, int cnt)
873*4882a593Smuzhiyun {
874*4882a593Smuzhiyun struct ufs_sb_private_info * uspi;
875*4882a593Smuzhiyun int i, start, end, forw, back;
876*4882a593Smuzhiyun
877*4882a593Smuzhiyun uspi = UFS_SB(sb)->s_uspi;
878*4882a593Smuzhiyun if (uspi->s_contigsumsize <= 0)
879*4882a593Smuzhiyun return;
880*4882a593Smuzhiyun
881*4882a593Smuzhiyun if (cnt > 0)
882*4882a593Smuzhiyun ubh_setbit(UCPI_UBH(ucpi), ucpi->c_clusteroff, blkno);
883*4882a593Smuzhiyun else
884*4882a593Smuzhiyun ubh_clrbit(UCPI_UBH(ucpi), ucpi->c_clusteroff, blkno);
885*4882a593Smuzhiyun
886*4882a593Smuzhiyun /*
887*4882a593Smuzhiyun * Find the size of the cluster going forward.
888*4882a593Smuzhiyun */
889*4882a593Smuzhiyun start = blkno + 1;
890*4882a593Smuzhiyun end = start + uspi->s_contigsumsize;
891*4882a593Smuzhiyun if ( end >= ucpi->c_nclusterblks)
892*4882a593Smuzhiyun end = ucpi->c_nclusterblks;
893*4882a593Smuzhiyun i = ubh_find_next_zero_bit (UCPI_UBH(ucpi), ucpi->c_clusteroff, end, start);
894*4882a593Smuzhiyun if (i > end)
895*4882a593Smuzhiyun i = end;
896*4882a593Smuzhiyun forw = i - start;
897*4882a593Smuzhiyun
898*4882a593Smuzhiyun /*
899*4882a593Smuzhiyun * Find the size of the cluster going backward.
900*4882a593Smuzhiyun */
901*4882a593Smuzhiyun start = blkno - 1;
902*4882a593Smuzhiyun end = start - uspi->s_contigsumsize;
903*4882a593Smuzhiyun if (end < 0 )
904*4882a593Smuzhiyun end = -1;
905*4882a593Smuzhiyun i = ubh_find_last_zero_bit (UCPI_UBH(ucpi), ucpi->c_clusteroff, start, end);
906*4882a593Smuzhiyun if ( i < end)
907*4882a593Smuzhiyun i = end;
908*4882a593Smuzhiyun back = start - i;
909*4882a593Smuzhiyun
910*4882a593Smuzhiyun /*
911*4882a593Smuzhiyun * Account for old cluster and the possibly new forward and
912*4882a593Smuzhiyun * back clusters.
913*4882a593Smuzhiyun */
914*4882a593Smuzhiyun i = back + forw + 1;
915*4882a593Smuzhiyun if (i > uspi->s_contigsumsize)
916*4882a593Smuzhiyun i = uspi->s_contigsumsize;
917*4882a593Smuzhiyun fs32_add(sb, (__fs32*)ubh_get_addr(UCPI_UBH(ucpi), ucpi->c_clustersumoff + (i << 2)), cnt);
918*4882a593Smuzhiyun if (back > 0)
919*4882a593Smuzhiyun fs32_sub(sb, (__fs32*)ubh_get_addr(UCPI_UBH(ucpi), ucpi->c_clustersumoff + (back << 2)), cnt);
920*4882a593Smuzhiyun if (forw > 0)
921*4882a593Smuzhiyun fs32_sub(sb, (__fs32*)ubh_get_addr(UCPI_UBH(ucpi), ucpi->c_clustersumoff + (forw << 2)), cnt);
922*4882a593Smuzhiyun }
923*4882a593Smuzhiyun
924*4882a593Smuzhiyun
925*4882a593Smuzhiyun static unsigned char ufs_fragtable_8fpb[] = {
926*4882a593Smuzhiyun 0x00, 0x01, 0x01, 0x02, 0x01, 0x01, 0x02, 0x04, 0x01, 0x01, 0x01, 0x03, 0x02, 0x03, 0x04, 0x08,
927*4882a593Smuzhiyun 0x01, 0x01, 0x01, 0x03, 0x01, 0x01, 0x03, 0x05, 0x02, 0x03, 0x03, 0x02, 0x04, 0x05, 0x08, 0x10,
928*4882a593Smuzhiyun 0x01, 0x01, 0x01, 0x03, 0x01, 0x01, 0x03, 0x05, 0x01, 0x01, 0x01, 0x03, 0x03, 0x03, 0x05, 0x09,
929*4882a593Smuzhiyun 0x02, 0x03, 0x03, 0x02, 0x03, 0x03, 0x02, 0x06, 0x04, 0x05, 0x05, 0x06, 0x08, 0x09, 0x10, 0x20,
930*4882a593Smuzhiyun 0x01, 0x01, 0x01, 0x03, 0x01, 0x01, 0x03, 0x05, 0x01, 0x01, 0x01, 0x03, 0x03, 0x03, 0x05, 0x09,
931*4882a593Smuzhiyun 0x01, 0x01, 0x01, 0x03, 0x01, 0x01, 0x03, 0x05, 0x03, 0x03, 0x03, 0x03, 0x05, 0x05, 0x09, 0x11,
932*4882a593Smuzhiyun 0x02, 0x03, 0x03, 0x02, 0x03, 0x03, 0x02, 0x06, 0x03, 0x03, 0x03, 0x03, 0x02, 0x03, 0x06, 0x0A,
933*4882a593Smuzhiyun 0x04, 0x05, 0x05, 0x06, 0x05, 0x05, 0x06, 0x04, 0x08, 0x09, 0x09, 0x0A, 0x10, 0x11, 0x20, 0x40,
934*4882a593Smuzhiyun 0x01, 0x01, 0x01, 0x03, 0x01, 0x01, 0x03, 0x05, 0x01, 0x01, 0x01, 0x03, 0x03, 0x03, 0x05, 0x09,
935*4882a593Smuzhiyun 0x01, 0x01, 0x01, 0x03, 0x01, 0x01, 0x03, 0x05, 0x03, 0x03, 0x03, 0x03, 0x05, 0x05, 0x09, 0x11,
936*4882a593Smuzhiyun 0x01, 0x01, 0x01, 0x03, 0x01, 0x01, 0x03, 0x05, 0x01, 0x01, 0x01, 0x03, 0x03, 0x03, 0x05, 0x09,
937*4882a593Smuzhiyun 0x03, 0x03, 0x03, 0x03, 0x03, 0x03, 0x03, 0x07, 0x05, 0x05, 0x05, 0x07, 0x09, 0x09, 0x11, 0x21,
938*4882a593Smuzhiyun 0x02, 0x03, 0x03, 0x02, 0x03, 0x03, 0x02, 0x06, 0x03, 0x03, 0x03, 0x03, 0x02, 0x03, 0x06, 0x0A,
939*4882a593Smuzhiyun 0x03, 0x03, 0x03, 0x03, 0x03, 0x03, 0x03, 0x07, 0x02, 0x03, 0x03, 0x02, 0x06, 0x07, 0x0A, 0x12,
940*4882a593Smuzhiyun 0x04, 0x05, 0x05, 0x06, 0x05, 0x05, 0x06, 0x04, 0x05, 0x05, 0x05, 0x07, 0x06, 0x07, 0x04, 0x0C,
941*4882a593Smuzhiyun 0x08, 0x09, 0x09, 0x0A, 0x09, 0x09, 0x0A, 0x0C, 0x10, 0x11, 0x11, 0x12, 0x20, 0x21, 0x40, 0x80,
942*4882a593Smuzhiyun };
943*4882a593Smuzhiyun
944*4882a593Smuzhiyun static unsigned char ufs_fragtable_other[] = {
945*4882a593Smuzhiyun 0x00, 0x16, 0x16, 0x2A, 0x16, 0x16, 0x26, 0x4E, 0x16, 0x16, 0x16, 0x3E, 0x2A, 0x3E, 0x4E, 0x8A,
946*4882a593Smuzhiyun 0x16, 0x16, 0x16, 0x3E, 0x16, 0x16, 0x36, 0x5E, 0x16, 0x16, 0x16, 0x3E, 0x3E, 0x3E, 0x5E, 0x9E,
947*4882a593Smuzhiyun 0x16, 0x16, 0x16, 0x3E, 0x16, 0x16, 0x36, 0x5E, 0x16, 0x16, 0x16, 0x3E, 0x3E, 0x3E, 0x5E, 0x9E,
948*4882a593Smuzhiyun 0x2A, 0x3E, 0x3E, 0x2A, 0x3E, 0x3E, 0x2E, 0x6E, 0x3E, 0x3E, 0x3E, 0x3E, 0x2A, 0x3E, 0x6E, 0xAA,
949*4882a593Smuzhiyun 0x16, 0x16, 0x16, 0x3E, 0x16, 0x16, 0x36, 0x5E, 0x16, 0x16, 0x16, 0x3E, 0x3E, 0x3E, 0x5E, 0x9E,
950*4882a593Smuzhiyun 0x16, 0x16, 0x16, 0x3E, 0x16, 0x16, 0x36, 0x5E, 0x16, 0x16, 0x16, 0x3E, 0x3E, 0x3E, 0x5E, 0x9E,
951*4882a593Smuzhiyun 0x26, 0x36, 0x36, 0x2E, 0x36, 0x36, 0x26, 0x6E, 0x36, 0x36, 0x36, 0x3E, 0x2E, 0x3E, 0x6E, 0xAE,
952*4882a593Smuzhiyun 0x4E, 0x5E, 0x5E, 0x6E, 0x5E, 0x5E, 0x6E, 0x4E, 0x5E, 0x5E, 0x5E, 0x7E, 0x6E, 0x7E, 0x4E, 0xCE,
953*4882a593Smuzhiyun 0x16, 0x16, 0x16, 0x3E, 0x16, 0x16, 0x36, 0x5E, 0x16, 0x16, 0x16, 0x3E, 0x3E, 0x3E, 0x5E, 0x9E,
954*4882a593Smuzhiyun 0x16, 0x16, 0x16, 0x3E, 0x16, 0x16, 0x36, 0x5E, 0x16, 0x16, 0x16, 0x3E, 0x3E, 0x3E, 0x5E, 0x9E,
955*4882a593Smuzhiyun 0x16, 0x16, 0x16, 0x3E, 0x16, 0x16, 0x36, 0x5E, 0x16, 0x16, 0x16, 0x3E, 0x3E, 0x3E, 0x5E, 0x9E,
956*4882a593Smuzhiyun 0x3E, 0x3E, 0x3E, 0x3E, 0x3E, 0x3E, 0x3E, 0x7E, 0x3E, 0x3E, 0x3E, 0x3E, 0x3E, 0x3E, 0x7E, 0xBE,
957*4882a593Smuzhiyun 0x2A, 0x3E, 0x3E, 0x2A, 0x3E, 0x3E, 0x2E, 0x6E, 0x3E, 0x3E, 0x3E, 0x3E, 0x2A, 0x3E, 0x6E, 0xAA,
958*4882a593Smuzhiyun 0x3E, 0x3E, 0x3E, 0x3E, 0x3E, 0x3E, 0x3E, 0x7E, 0x3E, 0x3E, 0x3E, 0x3E, 0x3E, 0x3E, 0x7E, 0xBE,
959*4882a593Smuzhiyun 0x4E, 0x5E, 0x5E, 0x6E, 0x5E, 0x5E, 0x6E, 0x4E, 0x5E, 0x5E, 0x5E, 0x7E, 0x6E, 0x7E, 0x4E, 0xCE,
960*4882a593Smuzhiyun 0x8A, 0x9E, 0x9E, 0xAA, 0x9E, 0x9E, 0xAE, 0xCE, 0x9E, 0x9E, 0x9E, 0xBE, 0xAA, 0xBE, 0xCE, 0x8A,
961*4882a593Smuzhiyun };
962