18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
28c2ecf20Sopenharmony_ci/*
38c2ecf20Sopenharmony_ci * QNX4 file system, Linux implementation.
48c2ecf20Sopenharmony_ci *
58c2ecf20Sopenharmony_ci * Version : 0.2.1
68c2ecf20Sopenharmony_ci *
78c2ecf20Sopenharmony_ci * Using parts of the xiafs filesystem.
88c2ecf20Sopenharmony_ci *
98c2ecf20Sopenharmony_ci * History :
108c2ecf20Sopenharmony_ci *
118c2ecf20Sopenharmony_ci * 28-05-1998 by Richard Frowijn : first release.
128c2ecf20Sopenharmony_ci * 20-06-1998 by Frank Denis : basic optimisations.
138c2ecf20Sopenharmony_ci * 25-06-1998 by Frank Denis : qnx4_is_free, qnx4_set_bitmap, qnx4_bmap .
148c2ecf20Sopenharmony_ci * 28-06-1998 by Frank Denis : qnx4_free_inode (to be fixed) .
158c2ecf20Sopenharmony_ci */
168c2ecf20Sopenharmony_ci
178c2ecf20Sopenharmony_ci#include <linux/buffer_head.h>
188c2ecf20Sopenharmony_ci#include <linux/bitops.h>
198c2ecf20Sopenharmony_ci#include "qnx4.h"
208c2ecf20Sopenharmony_ci
218c2ecf20Sopenharmony_ciunsigned long qnx4_count_free_blocks(struct super_block *sb)
228c2ecf20Sopenharmony_ci{
238c2ecf20Sopenharmony_ci	int start = le32_to_cpu(qnx4_sb(sb)->BitMap->di_first_xtnt.xtnt_blk) - 1;
248c2ecf20Sopenharmony_ci	int total = 0;
258c2ecf20Sopenharmony_ci	int total_free = 0;
268c2ecf20Sopenharmony_ci	int offset = 0;
278c2ecf20Sopenharmony_ci	int size = le32_to_cpu(qnx4_sb(sb)->BitMap->di_size);
288c2ecf20Sopenharmony_ci	struct buffer_head *bh;
298c2ecf20Sopenharmony_ci
308c2ecf20Sopenharmony_ci	while (total < size) {
318c2ecf20Sopenharmony_ci		int bytes = min(size - total, QNX4_BLOCK_SIZE);
328c2ecf20Sopenharmony_ci
338c2ecf20Sopenharmony_ci		if ((bh = sb_bread(sb, start + offset)) == NULL) {
348c2ecf20Sopenharmony_ci			printk(KERN_ERR "qnx4: I/O error in counting free blocks\n");
358c2ecf20Sopenharmony_ci			break;
368c2ecf20Sopenharmony_ci		}
378c2ecf20Sopenharmony_ci		total_free += bytes * BITS_PER_BYTE -
388c2ecf20Sopenharmony_ci				memweight(bh->b_data, bytes);
398c2ecf20Sopenharmony_ci		brelse(bh);
408c2ecf20Sopenharmony_ci		total += bytes;
418c2ecf20Sopenharmony_ci		offset++;
428c2ecf20Sopenharmony_ci	}
438c2ecf20Sopenharmony_ci
448c2ecf20Sopenharmony_ci	return total_free;
458c2ecf20Sopenharmony_ci}
46