162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * QNX4 file system, Linux implementation.
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Version : 0.2.1
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci * Using parts of the xiafs filesystem.
862306a36Sopenharmony_ci *
962306a36Sopenharmony_ci * History :
1062306a36Sopenharmony_ci *
1162306a36Sopenharmony_ci * 28-05-1998 by Richard Frowijn : first release.
1262306a36Sopenharmony_ci * 20-06-1998 by Frank Denis : basic optimisations.
1362306a36Sopenharmony_ci * 25-06-1998 by Frank Denis : qnx4_is_free, qnx4_set_bitmap, qnx4_bmap .
1462306a36Sopenharmony_ci * 28-06-1998 by Frank Denis : qnx4_free_inode (to be fixed) .
1562306a36Sopenharmony_ci */
1662306a36Sopenharmony_ci
1762306a36Sopenharmony_ci#include <linux/buffer_head.h>
1862306a36Sopenharmony_ci#include <linux/bitops.h>
1962306a36Sopenharmony_ci#include "qnx4.h"
2062306a36Sopenharmony_ci
2162306a36Sopenharmony_ciunsigned long qnx4_count_free_blocks(struct super_block *sb)
2262306a36Sopenharmony_ci{
2362306a36Sopenharmony_ci	int start = le32_to_cpu(qnx4_sb(sb)->BitMap->di_first_xtnt.xtnt_blk) - 1;
2462306a36Sopenharmony_ci	int total = 0;
2562306a36Sopenharmony_ci	int total_free = 0;
2662306a36Sopenharmony_ci	int offset = 0;
2762306a36Sopenharmony_ci	int size = le32_to_cpu(qnx4_sb(sb)->BitMap->di_size);
2862306a36Sopenharmony_ci	struct buffer_head *bh;
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_ci	while (total < size) {
3162306a36Sopenharmony_ci		int bytes = min(size - total, QNX4_BLOCK_SIZE);
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_ci		if ((bh = sb_bread(sb, start + offset)) == NULL) {
3462306a36Sopenharmony_ci			printk(KERN_ERR "qnx4: I/O error in counting free blocks\n");
3562306a36Sopenharmony_ci			break;
3662306a36Sopenharmony_ci		}
3762306a36Sopenharmony_ci		total_free += bytes * BITS_PER_BYTE -
3862306a36Sopenharmony_ci				memweight(bh->b_data, bytes);
3962306a36Sopenharmony_ci		brelse(bh);
4062306a36Sopenharmony_ci		total += bytes;
4162306a36Sopenharmony_ci		offset++;
4262306a36Sopenharmony_ci	}
4362306a36Sopenharmony_ci
4462306a36Sopenharmony_ci	return total_free;
4562306a36Sopenharmony_ci}
46