[Ocfs2-tools-devel] [PATCH 2/3] O2info: O2info: Add new running codes for '--freeinode'

Tristan Ye tristan.ye at oracle.com
Wed Nov 3 04:08:59 PDT 2010


Patch teaches o2info to dump fs's free inode space among nodes.

Task of '--freeinode' will also be capable of two approaches,
including libocfs2 and ioctl solutions.

Signed-off-by: Tristan Ye <tristan.ye at oracle.com>
---
 include/ocfs2-kernel/ocfs2_ioctl.h |   11 +++++
 o2info/libo2info.c                 |   58 +++++++++++++++++++++++++++
 o2info/libo2info.h                 |   11 +++++
 o2info/o2info.1.in                 |    6 ++-
 o2info/o2info.c                    |   15 +++++++
 o2info/operations.c                |   75 ++++++++++++++++++++++++++++++++++++
 6 files changed, 175 insertions(+), 1 deletions(-)

diff --git a/include/ocfs2-kernel/ocfs2_ioctl.h b/include/ocfs2-kernel/ocfs2_ioctl.h
index 9bc5354..57aa718 100644
--- a/include/ocfs2-kernel/ocfs2_ioctl.h
+++ b/include/ocfs2-kernel/ocfs2_ioctl.h
@@ -142,6 +142,16 @@ struct ocfs2_info_journal_size {
 	__u64 ij_journal_size;
 };
 
+struct ocfs2_info_freeinode {
+	struct ocfs2_info_request ifi_req;
+	struct ocfs2_info_local_freeinode {
+		__u64 lfi_total;
+		__u64 lfi_free;
+	} ifi_stat[OCFS2_MAX_SLOTS];
+	__u32 ifi_slotnum; /* out */
+	__u32 ifi_pad;
+};
+
 /* Codes for ocfs2_info_request */
 enum ocfs2_info_type {
 	OCFS2_INFO_CLUSTERSIZE = 1,
@@ -151,6 +161,7 @@ enum ocfs2_info_type {
 	OCFS2_INFO_UUID,
 	OCFS2_INFO_FS_FEATURES,
 	OCFS2_INFO_JOURNAL_SIZE,
+	OCFS2_INFO_FREEINODE,
 	OCFS2_INFO_NUM_TYPES
 };
 
diff --git a/o2info/libo2info.c b/o2info/libo2info.c
index 6db04d0..912c938 100644
--- a/o2info/libo2info.c
+++ b/o2info/libo2info.c
@@ -21,6 +21,8 @@
 #define _LARGEFILE64_SOURCE
 #define _GNU_SOURCE
 
+#include <inttypes.h>
+
 #include "ocfs2/ocfs2.h"
 #include "tools-internal/verbose.h"
 #include "libo2info.h"
@@ -95,3 +97,59 @@ out:
 
 	return err;
 }
+
+int o2info_get_freeinode(ocfs2_filesys *fs, struct o2info_freeinode *ofi)
+{
+
+	int ret = 0, i, j;
+	char *block = NULL;
+	uint64_t inode_alloc;
+
+	struct ocfs2_dinode *dinode_alloc = NULL;
+	struct ocfs2_chain_list *cl = NULL;
+	struct ocfs2_chain_rec *rec = NULL;
+	struct ocfs2_super_block *sb = OCFS2_RAW_SB(fs->fs_super);
+
+	ofi->slotnum = sb->s_max_slots;
+
+	ret = ocfs2_malloc_block(fs->fs_io, &block);
+	if (ret) {
+		tcom_err(ret, "while allocating block buffer");
+		goto out;
+	}
+
+	dinode_alloc = (struct ocfs2_dinode *)block;
+
+	for (i = 0; i < ofi->slotnum; i++) {
+
+		ofi->fi[i].total = ofi->fi[i].free = 0;
+
+		ret = ocfs2_lookup_system_inode(fs, INODE_ALLOC_SYSTEM_INODE,
+						i, &inode_alloc);
+		if (ret) {
+			tcom_err(ret, "while looking up the global"
+				 " bitmap inode");
+			goto out;
+		}
+
+		ret = ocfs2_read_inode(fs, inode_alloc, (char *)dinode_alloc);
+		if (ret) {
+			tcom_err(ret, "reading global_bitmap inode "
+				 "%"PRIu64" for stats", inode_alloc);
+			goto out;
+		}
+
+		cl = &(dinode_alloc->id2.i_chain);
+
+		for (j = 0; j < cl->cl_next_free_rec; j++) {
+			rec = &(cl->cl_recs[j]);
+			ofi->fi[i].total += rec->c_total;
+			ofi->fi[i].free += rec->c_free;
+		}
+	}
+out:
+	if (block)
+		ocfs2_free(&block);
+
+	return ret;
+}
diff --git a/o2info/libo2info.h b/o2info/libo2info.h
index 081eace..08a0500 100644
--- a/o2info/libo2info.h
+++ b/o2info/libo2info.h
@@ -40,8 +40,19 @@ struct o2info_mkfs {
 	uint64_t journal_size;
 };
 
+struct o2info_local_freeinode {
+	unsigned long total;
+	unsigned long free;
+};
+
+struct o2info_freeinode {
+	int slotnum;
+	struct o2info_local_freeinode fi[OCFS2_MAX_SLOTS];
+};
+
 int o2info_get_fs_features(ocfs2_filesys *fs, struct o2info_fs_features *ofs);
 int o2info_get_volinfo(ocfs2_filesys *fs, struct o2info_volinfo *vf);
 int o2info_get_mkfs(ocfs2_filesys *fs, struct o2info_mkfs *oms);
+int o2info_get_freeinode(ocfs2_filesys *fs, struct o2info_freeinode *ofi);
 
 #endif
diff --git a/o2info/o2info.1.in b/o2info/o2info.1.in
index 3ddfba6..2c77457 100644
--- a/o2info/o2info.1.in
+++ b/o2info/o2info.1.in
@@ -2,7 +2,7 @@
 .SH "NAME"
 o2info \- Dump \fIOCFS2\fR file system information on disk.
 .SH "SYNOPSIS"
-\fBo2info\fR [\fB\-C|\-\-cluster\-coherent\fR] [\fB\-\-fs\-features\fR] [\fB\-\-volinfo\fR] [\fB\-\-mkfs\fR] <\fBdevice or file\fR>
+\fBo2info\fR [\fB\-C|\-\-cluster\-coherent\fR] [\fB\-\-fs\-features\fR] [\fB\-\-volinfo\fR] [\fB\-\-mkfs\fR] [\fB\-\-freeinode\fR] <\fBdevice or file\fR>
 
 .SH "DESCRIPTION"
 .PP
@@ -27,6 +27,10 @@ Dump an original format of mkfs.ocfs's arguments, importantly, which can be used
 #mkfs.ocfs2 "$(o2info --mkfs /dev/name/or/path/of/file)" /dev/sdaX
 
 .TP
+\fB\-\-freeinode\fR
+Display free inode space for all nodes in \fIOCFS2\fR cluster.
+
+.TP
 \fB\-V, \-\-version\fR
 Show version and exit.
 
diff --git a/o2info/o2info.c b/o2info/o2info.c
index 9e23cfb..62a92f1 100644
--- a/o2info/o2info.c
+++ b/o2info/o2info.c
@@ -37,6 +37,7 @@
 extern struct o2info_operation fs_features_op;
 extern struct o2info_operation volinfo_op;
 extern struct o2info_operation mkfs_op;
+extern struct o2info_operation freeinode_op;
 
 static LIST_HEAD(o2info_op_task_list);
 static int o2info_op_task_count;
@@ -141,6 +142,19 @@ static struct o2info_option mkfs_option = {
 	.opt_private	= NULL,
 };
 
+static struct o2info_option freeinode_option = {
+	.opt_option	= {
+		.name		= "freeinode",
+		.val		= CHAR_MAX,
+		.has_arg	= 0,
+		.flag		= NULL,
+	},
+	.opt_help	= "   --freeinode",
+	.opt_handler	= NULL,
+	.opt_op		= &freeinode_op,
+	.opt_private	= NULL,
+};
+
 static struct o2info_option *options[] = {
 	&help_option,
 	&version_option,
@@ -148,6 +162,7 @@ static struct o2info_option *options[] = {
 	&fs_features_option,
 	&volinfo_option,
 	&mkfs_option,
+	&freeinode_option,
 	NULL,
 };
 
diff --git a/o2info/operations.c b/o2info/operations.c
index f7216d8..a0e85d6 100644
--- a/o2info/operations.c
+++ b/o2info/operations.c
@@ -537,3 +537,78 @@ out:
 DEFINE_O2INFO_OP(mkfs,
 		 mkfs_run,
 		 NULL);
+
+static int get_freeinode_ioctl(struct o2info_operation *op,
+			       int fd,
+			       struct o2info_freeinode *ofi)
+{
+	uint64_t reqs[1];
+	int ret = 0, flags = 0;
+	struct ocfs2_info info;
+	struct ocfs2_info_freeinode oifi;
+	uint32_t unknowns = 0, errors = 0, fills = 0;
+
+	memset(ofi, 0, sizeof(*ofi));
+
+	if (!cluster_coherent)
+		flags |= OCFS2_INFO_FL_NON_COHERENT;
+
+	o2info_fill_request((struct ocfs2_info_request *)&oifi, sizeof(oifi),
+			    OCFS2_INFO_FREEINODE, flags);
+
+	reqs[0] = (unsigned long)&oifi;
+
+	info.oi_requests = (uint64_t)reqs;
+	info.oi_count = 1;
+
+	ret = ioctl(fd, OCFS2_IOC_INFO, &info);
+	if (ret) {
+		ret = errno;
+		o2i_error(op, "ioctl failed: %s\n", strerror(ret));
+		o2i_scan_requests(op, info, &unknowns, &errors, &fills);
+		goto out;
+	}
+
+	if (oifi.ifi_req.ir_flags & OCFS2_INFO_FL_FILLED) {
+		ofi->slotnum = oifi.ifi_slotnum;
+		memcpy(ofi->fi, oifi.ifi_stat,
+		       sizeof(struct o2info_local_freeinode) * OCFS2_MAX_SLOTS);
+	}
+
+out:
+	return ret;
+}
+
+static int freeinode_run(struct o2info_operation *op,
+			struct o2info_method *om,
+			void *arg)
+{
+	int ret = 0, i;
+
+	struct o2info_freeinode ofi;
+	unsigned long total = 0, free = 0;
+
+	if (om->om_method == O2INFO_USE_IOCTL)
+		ret = get_freeinode_ioctl(op, om->om_fd, &ofi);
+	else
+		ret = o2info_get_freeinode(om->om_fs, &ofi);
+
+	if (ret)
+		return ret;
+
+	fprintf(stdout, "Slot\t\tSpace\t\tFree\n");
+	for (i = 0; i < ofi.slotnum ; i++) {
+		fprintf(stdout, "%3d\t%13lu\t%12lu\n", i, ofi.fi[i].total,
+			ofi.fi[i].free);
+		total += ofi.fi[i].total;
+		free += ofi.fi[i].free;
+	}
+
+	fprintf(stdout, "Total\t%13lu\t%12lu\n", total, free);
+
+	return ret;
+}
+
+DEFINE_O2INFO_OP(freeinode,
+		 freeinode_run,
+		 NULL);
-- 
1.5.5




More information about the Ocfs2-tools-devel mailing list