Blame view

fs/hfsplus/brec.c 13.6 KB
b24413180   Greg Kroah-Hartman   License cleanup: ...
1
  // SPDX-License-Identifier: GPL-2.0
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
  /*
   *  linux/fs/hfsplus/brec.c
   *
   * Copyright (C) 2001
   * Brad Boyer (flar@allandria.com)
   * (C) 2003 Ardis Technologies <roman@ardistech.com>
   *
   * Handle individual btree records
   */
  
  #include "hfsplus_fs.h"
  #include "hfsplus_raw.h"
  
  static struct hfs_bnode *hfs_bnode_split(struct hfs_find_data *fd);
  static int hfs_brec_update_parent(struct hfs_find_data *fd);
  static int hfs_btree_inc_height(struct hfs_btree *);
  
  /* Get the length and offset of the given record in the given node */
  u16 hfs_brec_lenoff(struct hfs_bnode *node, u16 rec, u16 *off)
  {
  	__be16 retval[2];
  	u16 dataoff;
  
  	dataoff = node->tree->node_size - (rec + 2) * 2;
  	hfs_bnode_read(node, retval, dataoff, 4);
  	*off = be16_to_cpu(retval[1]);
  	return be16_to_cpu(retval[0]) - *off;
  }
  
  /* Get the length of the key from a keyed record */
  u16 hfs_brec_keylen(struct hfs_bnode *node, u16 rec)
  {
  	u16 retval, recoff;
  
  	if (node->type != HFS_NODE_INDEX && node->type != HFS_NODE_LEAF)
  		return 0;
  
  	if ((node->type == HFS_NODE_INDEX) &&
324ef39a8   Vyacheslav Dubeyko   hfsplus: add supp...
40
41
  	   !(node->tree->attributes & HFS_TREE_VARIDXKEYS) &&
  	   (node->tree->cnid != HFSPLUS_ATTR_CNID)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
42
43
  		retval = node->tree->max_key_len + 2;
  	} else {
2753cc281   Anton Salikhmetov   hfsplus: over 80 ...
44
45
  		recoff = hfs_bnode_read_u16(node,
  			node->tree->node_size - (rec + 1) * 2);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
46
47
  		if (!recoff)
  			return 0;
aac4e4198   Naohiro Aota   hfsplus: Add addi...
48
  		if (recoff > node->tree->node_size - 2) {
d61426732   Joe Perches   hfs/hfsplus: conv...
49
50
  			pr_err("recoff %d too large
  ", recoff);
aac4e4198   Naohiro Aota   hfsplus: Add addi...
51
52
  			return 0;
  		}
13571a697   Christoph Hellwig   hfsplus: validate...
53
54
55
  
  		retval = hfs_bnode_read_u16(node, recoff) + 2;
  		if (retval > node->tree->max_key_len + 2) {
d61426732   Joe Perches   hfs/hfsplus: conv...
56
57
  			pr_err("keylen %d too large
  ",
13571a697   Christoph Hellwig   hfsplus: validate...
58
59
  				retval);
  			retval = 0;
9250f9259   Eric Sandeen   hfsplus: handle m...
60
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
  	}
  	return retval;
  }
  
  int hfs_brec_insert(struct hfs_find_data *fd, void *entry, int entry_len)
  {
  	struct hfs_btree *tree;
  	struct hfs_bnode *node, *new_node;
  	int size, key_len, rec;
  	int data_off, end_off;
  	int idx_rec_off, data_rec_off, end_rec_off;
  	__be32 cnid;
  
  	tree = fd->tree;
  	if (!fd->bnode) {
  		if (!tree->root)
  			hfs_btree_inc_height(tree);
aba93a92f   Ernesto A. Fernandez   hfsplus: prevent ...
78
79
80
81
  		node = hfs_bnode_find(tree, tree->leaf_head);
  		if (IS_ERR(node))
  			return PTR_ERR(node);
  		fd->bnode = node;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
  		fd->record = -1;
  	}
  	new_node = NULL;
  	key_len = be16_to_cpu(fd->search_key->key_len) + 2;
  again:
  	/* new record idx and complete record size */
  	rec = fd->record + 1;
  	size = key_len + entry_len;
  
  	node = fd->bnode;
  	hfs_bnode_dump(node);
  	/* get last offset */
  	end_rec_off = tree->node_size - (node->num_recs + 1) * 2;
  	end_off = hfs_bnode_read_u16(node, end_rec_off);
  	end_rec_off -= 2;
c2b3e1f76   Joe Perches   hfs/hfsplus: conv...
97
98
  	hfs_dbg(BNODE_MOD, "insert_rec: %d, %d, %d, %d
  ",
2753cc281   Anton Salikhmetov   hfsplus: over 80 ...
99
  		rec, size, end_off, end_rec_off);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
  	if (size > end_rec_off - end_off) {
  		if (new_node)
  			panic("not enough room!
  ");
  		new_node = hfs_bnode_split(fd);
  		if (IS_ERR(new_node))
  			return PTR_ERR(new_node);
  		goto again;
  	}
  	if (node->type == HFS_NODE_LEAF) {
  		tree->leaf_count++;
  		mark_inode_dirty(tree->inode);
  	}
  	node->num_recs++;
  	/* write new last offset */
2753cc281   Anton Salikhmetov   hfsplus: over 80 ...
115
116
117
  	hfs_bnode_write_u16(node,
  		offsetof(struct hfs_bnode_desc, num_recs),
  		node->num_recs);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
  	hfs_bnode_write_u16(node, end_rec_off, end_off + size);
  	data_off = end_off;
  	data_rec_off = end_rec_off + 2;
  	idx_rec_off = tree->node_size - (rec + 1) * 2;
  	if (idx_rec_off == data_rec_off)
  		goto skip;
  	/* move all following entries */
  	do {
  		data_off = hfs_bnode_read_u16(node, data_rec_off + 2);
  		hfs_bnode_write_u16(node, data_rec_off, data_off + size);
  		data_rec_off += 2;
  	} while (data_rec_off < idx_rec_off);
  
  	/* move data away */
  	hfs_bnode_move(node, data_off + size, data_off,
  		       end_off - data_off);
  
  skip:
  	hfs_bnode_write(node, fd->search_key, data_off, key_len);
  	hfs_bnode_write(node, entry, data_off + key_len, entry_len);
  	hfs_bnode_dump(node);
98cf21c61   Sergei Antonov   hfsplus: fix B-tr...
139
140
141
142
143
144
145
146
  	/*
  	 * update parent key if we inserted a key
  	 * at the start of the node and it is not the new node
  	 */
  	if (!rec && new_node != node) {
  		hfs_bnode_read_key(node, fd->search_key, data_off + size);
  		hfs_brec_update_parent(fd);
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
147

98cf21c61   Sergei Antonov   hfsplus: fix B-tr...
148
  	if (new_node) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
149
150
151
152
153
154
155
156
157
158
159
160
161
162
  		hfs_bnode_put(fd->bnode);
  		if (!new_node->parent) {
  			hfs_btree_inc_height(tree);
  			new_node->parent = tree->root;
  		}
  		fd->bnode = hfs_bnode_find(tree, new_node->parent);
  
  		/* create index data entry */
  		cnid = cpu_to_be32(new_node->this);
  		entry = &cnid;
  		entry_len = sizeof(cnid);
  
  		/* get index key */
  		hfs_bnode_read_key(new_node, fd->search_key, 14);
324ef39a8   Vyacheslav Dubeyko   hfsplus: add supp...
163
  		__hfs_brec_find(fd->bnode, fd, hfs_find_rec_by_key);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
164
165
166
  
  		hfs_bnode_put(new_node);
  		new_node = NULL;
324ef39a8   Vyacheslav Dubeyko   hfsplus: add supp...
167
168
  		if ((tree->attributes & HFS_TREE_VARIDXKEYS) ||
  				(tree->cnid == HFSPLUS_ATTR_CNID))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
169
170
  			key_len = be16_to_cpu(fd->search_key->key_len) + 2;
  		else {
2753cc281   Anton Salikhmetov   hfsplus: over 80 ...
171
172
  			fd->search_key->key_len =
  				cpu_to_be16(tree->max_key_len);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
173
174
175
176
  			key_len = tree->max_key_len + 2;
  		}
  		goto again;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
  	return 0;
  }
  
  int hfs_brec_remove(struct hfs_find_data *fd)
  {
  	struct hfs_btree *tree;
  	struct hfs_bnode *node, *parent;
  	int end_off, rec_off, data_off, size;
  
  	tree = fd->tree;
  	node = fd->bnode;
  again:
  	rec_off = tree->node_size - (fd->record + 2) * 2;
  	end_off = tree->node_size - (node->num_recs + 1) * 2;
  
  	if (node->type == HFS_NODE_LEAF) {
  		tree->leaf_count--;
  		mark_inode_dirty(tree->inode);
  	}
  	hfs_bnode_dump(node);
c2b3e1f76   Joe Perches   hfs/hfsplus: conv...
197
198
  	hfs_dbg(BNODE_MOD, "remove_rec: %d, %d
  ",
2753cc281   Anton Salikhmetov   hfsplus: over 80 ...
199
  		fd->record, fd->keylength + fd->entrylength);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
200
201
202
203
204
205
206
207
208
  	if (!--node->num_recs) {
  		hfs_bnode_unlink(node);
  		if (!node->parent)
  			return 0;
  		parent = hfs_bnode_find(tree, node->parent);
  		if (IS_ERR(parent))
  			return PTR_ERR(parent);
  		hfs_bnode_put(node);
  		node = fd->bnode = parent;
324ef39a8   Vyacheslav Dubeyko   hfsplus: add supp...
209
  		__hfs_brec_find(node, fd, hfs_find_rec_by_key);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
210
211
  		goto again;
  	}
2753cc281   Anton Salikhmetov   hfsplus: over 80 ...
212
213
214
  	hfs_bnode_write_u16(node,
  		offsetof(struct hfs_bnode_desc, num_recs),
  		node->num_recs);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
  
  	if (rec_off == end_off)
  		goto skip;
  	size = fd->keylength + fd->entrylength;
  
  	do {
  		data_off = hfs_bnode_read_u16(node, rec_off);
  		hfs_bnode_write_u16(node, rec_off + 2, data_off - size);
  		rec_off -= 2;
  	} while (rec_off >= end_off);
  
  	/* fill hole */
  	hfs_bnode_move(node, fd->keyoffset, fd->keyoffset + size,
  		       data_off - fd->keyoffset - size);
  skip:
  	hfs_bnode_dump(node);
  	if (!fd->record)
  		hfs_brec_update_parent(fd);
  	return 0;
  }
  
  static struct hfs_bnode *hfs_bnode_split(struct hfs_find_data *fd)
  {
  	struct hfs_btree *tree;
b6b41424f   Al Viro   hfsplus: hfs_bnod...
239
  	struct hfs_bnode *node, *new_node, *next_node;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
240
241
242
243
244
245
246
247
248
249
  	struct hfs_bnode_desc node_desc;
  	int num_recs, new_rec_off, new_off, old_rec_off;
  	int data_start, data_end, size;
  
  	tree = fd->tree;
  	node = fd->bnode;
  	new_node = hfs_bmap_alloc(tree);
  	if (IS_ERR(new_node))
  		return new_node;
  	hfs_bnode_get(node);
c2b3e1f76   Joe Perches   hfs/hfsplus: conv...
250
251
  	hfs_dbg(BNODE_MOD, "split_nodes: %d - %d - %d
  ",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
252
253
254
255
256
257
  		node->this, new_node->this, node->next);
  	new_node->next = node->next;
  	new_node->prev = node->this;
  	new_node->parent = node->parent;
  	new_node->type = node->type;
  	new_node->height = node->height;
b6b41424f   Al Viro   hfsplus: hfs_bnod...
258
259
260
261
262
263
264
265
266
267
  	if (node->next)
  		next_node = hfs_bnode_find(tree, node->next);
  	else
  		next_node = NULL;
  
  	if (IS_ERR(next_node)) {
  		hfs_bnode_put(node);
  		hfs_bnode_put(new_node);
  		return next_node;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
268
269
270
271
272
273
274
275
276
277
278
279
280
  	size = tree->node_size / 2 - node->num_recs * 2 - 14;
  	old_rec_off = tree->node_size - 4;
  	num_recs = 1;
  	for (;;) {
  		data_start = hfs_bnode_read_u16(node, old_rec_off);
  		if (data_start > size)
  			break;
  		old_rec_off -= 2;
  		if (++num_recs < node->num_recs)
  			continue;
  		/* panic? */
  		hfs_bnode_put(node);
  		hfs_bnode_put(new_node);
b6b41424f   Al Viro   hfsplus: hfs_bnod...
281
282
  		if (next_node)
  			hfs_bnode_put(next_node);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
  		return ERR_PTR(-ENOSPC);
  	}
  
  	if (fd->record + 1 < num_recs) {
  		/* new record is in the lower half,
  		 * so leave some more space there
  		 */
  		old_rec_off += 2;
  		num_recs--;
  		data_start = hfs_bnode_read_u16(node, old_rec_off);
  	} else {
  		hfs_bnode_put(node);
  		hfs_bnode_get(new_node);
  		fd->bnode = new_node;
  		fd->record -= num_recs;
  		fd->keyoffset -= data_start - 14;
  		fd->entryoffset -= data_start - 14;
  	}
  	new_node->num_recs = node->num_recs - num_recs;
  	node->num_recs = num_recs;
  
  	new_rec_off = tree->node_size - 2;
  	new_off = 14;
  	size = data_start - new_off;
  	num_recs = new_node->num_recs;
  	data_end = data_start;
  	while (num_recs) {
  		hfs_bnode_write_u16(new_node, new_rec_off, new_off);
  		old_rec_off -= 2;
  		new_rec_off -= 2;
  		data_end = hfs_bnode_read_u16(node, old_rec_off);
  		new_off = data_end - size;
  		num_recs--;
  	}
  	hfs_bnode_write_u16(new_node, new_rec_off, new_off);
  	hfs_bnode_copy(new_node, 14, node, data_start, data_end - data_start);
  
  	/* update new bnode header */
  	node_desc.next = cpu_to_be32(new_node->next);
  	node_desc.prev = cpu_to_be32(new_node->prev);
  	node_desc.type = new_node->type;
  	node_desc.height = new_node->height;
  	node_desc.num_recs = cpu_to_be16(new_node->num_recs);
  	node_desc.reserved = 0;
  	hfs_bnode_write(new_node, &node_desc, 0, sizeof(node_desc));
  
  	/* update previous bnode header */
  	node->next = new_node->this;
  	hfs_bnode_read(node, &node_desc, 0, sizeof(node_desc));
  	node_desc.next = cpu_to_be32(node->next);
  	node_desc.num_recs = cpu_to_be16(node->num_recs);
  	hfs_bnode_write(node, &node_desc, 0, sizeof(node_desc));
  
  	/* update next bnode header */
b6b41424f   Al Viro   hfsplus: hfs_bnod...
337
  	if (next_node) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
  		next_node->prev = new_node->this;
  		hfs_bnode_read(next_node, &node_desc, 0, sizeof(node_desc));
  		node_desc.prev = cpu_to_be32(next_node->prev);
  		hfs_bnode_write(next_node, &node_desc, 0, sizeof(node_desc));
  		hfs_bnode_put(next_node);
  	} else if (node->this == tree->leaf_tail) {
  		/* if there is no next node, this might be the new tail */
  		tree->leaf_tail = new_node->this;
  		mark_inode_dirty(tree->inode);
  	}
  
  	hfs_bnode_dump(node);
  	hfs_bnode_dump(new_node);
  	hfs_bnode_put(node);
  
  	return new_node;
  }
  
  static int hfs_brec_update_parent(struct hfs_find_data *fd)
  {
  	struct hfs_btree *tree;
  	struct hfs_bnode *node, *new_node, *parent;
  	int newkeylen, diff;
  	int rec, rec_off, end_rec_off;
  	int start_off, end_off;
  
  	tree = fd->tree;
  	node = fd->bnode;
  	new_node = NULL;
  	if (!node->parent)
  		return 0;
  
  again:
  	parent = hfs_bnode_find(tree, node->parent);
  	if (IS_ERR(parent))
  		return PTR_ERR(parent);
324ef39a8   Vyacheslav Dubeyko   hfsplus: add supp...
374
  	__hfs_brec_find(parent, fd, hfs_find_rec_by_key);
98cf21c61   Sergei Antonov   hfsplus: fix B-tr...
375
376
  	if (fd->record < 0)
  		return -ENOENT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
377
378
379
380
  	hfs_bnode_dump(parent);
  	rec = fd->record;
  
  	/* size difference between old and new key */
324ef39a8   Vyacheslav Dubeyko   hfsplus: add supp...
381
382
  	if ((tree->attributes & HFS_TREE_VARIDXKEYS) ||
  				(tree->cnid == HFSPLUS_ATTR_CNID))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
383
384
385
  		newkeylen = hfs_bnode_read_u16(node, 14) + 2;
  	else
  		fd->keylength = newkeylen = tree->max_key_len + 2;
c2b3e1f76   Joe Perches   hfs/hfsplus: conv...
386
387
  	hfs_dbg(BNODE_MOD, "update_rec: %d, %d, %d
  ",
2753cc281   Anton Salikhmetov   hfsplus: over 80 ...
388
  		rec, fd->keylength, newkeylen);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
389
390
391
392
393
394
395
396
397
  
  	rec_off = tree->node_size - (rec + 2) * 2;
  	end_rec_off = tree->node_size - (parent->num_recs + 1) * 2;
  	diff = newkeylen - fd->keylength;
  	if (!diff)
  		goto skip;
  	if (diff > 0) {
  		end_off = hfs_bnode_read_u16(parent, end_rec_off);
  		if (end_rec_off - end_off < diff) {
c2b3e1f76   Joe Perches   hfs/hfsplus: conv...
398
399
  			hfs_dbg(BNODE_MOD, "splitting index node
  ");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
400
401
402
403
404
405
406
  			fd->bnode = parent;
  			new_node = hfs_bnode_split(fd);
  			if (IS_ERR(new_node))
  				return PTR_ERR(new_node);
  			parent = fd->bnode;
  			rec = fd->record;
  			rec_off = tree->node_size - (rec + 2) * 2;
2753cc281   Anton Salikhmetov   hfsplus: over 80 ...
407
408
  			end_rec_off = tree->node_size -
  				(parent->num_recs + 1) * 2;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
  		}
  	}
  
  	end_off = start_off = hfs_bnode_read_u16(parent, rec_off);
  	hfs_bnode_write_u16(parent, rec_off, start_off + diff);
  	start_off -= 4;	/* move previous cnid too */
  
  	while (rec_off > end_rec_off) {
  		rec_off -= 2;
  		end_off = hfs_bnode_read_u16(parent, rec_off);
  		hfs_bnode_write_u16(parent, rec_off, end_off + diff);
  	}
  	hfs_bnode_move(parent, start_off + diff, start_off,
  		       end_off - start_off);
  skip:
  	hfs_bnode_copy(parent, fd->keyoffset, node, 14, newkeylen);
  	hfs_bnode_dump(parent);
  
  	hfs_bnode_put(node);
  	node = parent;
  
  	if (new_node) {
  		__be32 cnid;
0a3021d4f   Ernesto A. Fernández   hfsplus: prevent ...
432
433
434
435
  		if (!new_node->parent) {
  			hfs_btree_inc_height(tree);
  			new_node->parent = tree->root;
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
436
437
438
439
  		fd->bnode = hfs_bnode_find(tree, new_node->parent);
  		/* create index key and entry */
  		hfs_bnode_read_key(new_node, fd->search_key, 14);
  		cnid = cpu_to_be32(new_node->this);
324ef39a8   Vyacheslav Dubeyko   hfsplus: add supp...
440
  		__hfs_brec_find(fd->bnode, fd, hfs_find_rec_by_key);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
441
442
443
444
445
446
447
448
449
450
  		hfs_brec_insert(fd, &cnid, sizeof(cnid));
  		hfs_bnode_put(fd->bnode);
  		hfs_bnode_put(new_node);
  
  		if (!rec) {
  			if (new_node == node)
  				goto out;
  			/* restore search_key */
  			hfs_bnode_read_key(node, fd->search_key, 14);
  		}
19a9d0f1a   Ernesto A. Fernández   hfsplus: fix BUG ...
451
  		new_node = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
  	}
  
  	if (!rec && node->parent)
  		goto again;
  out:
  	fd->bnode = node;
  	return 0;
  }
  
  static int hfs_btree_inc_height(struct hfs_btree *tree)
  {
  	struct hfs_bnode *node, *new_node;
  	struct hfs_bnode_desc node_desc;
  	int key_size, rec;
  	__be32 cnid;
  
  	node = NULL;
  	if (tree->root) {
  		node = hfs_bnode_find(tree, tree->root);
  		if (IS_ERR(node))
  			return PTR_ERR(node);
  	}
  	new_node = hfs_bmap_alloc(tree);
  	if (IS_ERR(new_node)) {
  		hfs_bnode_put(node);
  		return PTR_ERR(new_node);
  	}
  
  	tree->root = new_node->this;
  	if (!tree->depth) {
  		tree->leaf_head = tree->leaf_tail = new_node->this;
  		new_node->type = HFS_NODE_LEAF;
  		new_node->num_recs = 0;
  	} else {
  		new_node->type = HFS_NODE_INDEX;
  		new_node->num_recs = 1;
  	}
  	new_node->parent = 0;
  	new_node->next = 0;
  	new_node->prev = 0;
  	new_node->height = ++tree->depth;
  
  	node_desc.next = cpu_to_be32(new_node->next);
  	node_desc.prev = cpu_to_be32(new_node->prev);
  	node_desc.type = new_node->type;
  	node_desc.height = new_node->height;
  	node_desc.num_recs = cpu_to_be16(new_node->num_recs);
  	node_desc.reserved = 0;
  	hfs_bnode_write(new_node, &node_desc, 0, sizeof(node_desc));
  
  	rec = tree->node_size - 2;
  	hfs_bnode_write_u16(new_node, rec, 14);
  
  	if (node) {
  		/* insert old root idx into new root */
  		node->parent = tree->root;
  		if (node->type == HFS_NODE_LEAF ||
324ef39a8   Vyacheslav Dubeyko   hfsplus: add supp...
509
510
  				tree->attributes & HFS_TREE_VARIDXKEYS ||
  				tree->cnid == HFSPLUS_ATTR_CNID)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
511
512
513
514
  			key_size = hfs_bnode_read_u16(node, 14) + 2;
  		else
  			key_size = tree->max_key_len + 2;
  		hfs_bnode_copy(new_node, 14, node, 14, key_size);
324ef39a8   Vyacheslav Dubeyko   hfsplus: add supp...
515
516
  		if (!(tree->attributes & HFS_TREE_VARIDXKEYS) &&
  				(tree->cnid != HFSPLUS_ATTR_CNID)) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
  			key_size = tree->max_key_len + 2;
  			hfs_bnode_write_u16(new_node, 14, tree->max_key_len);
  		}
  		cnid = cpu_to_be32(node->this);
  		hfs_bnode_write(new_node, &cnid, 14 + key_size, 4);
  
  		rec -= 2;
  		hfs_bnode_write_u16(new_node, rec, 14 + key_size + 4);
  
  		hfs_bnode_put(node);
  	}
  	hfs_bnode_put(new_node);
  	mark_inode_dirty(tree->inode);
  
  	return 0;
  }