xref: /OK3568_Linux_fs/kernel/fs/ufs/balloc.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
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