chiark / gitweb /
libvolume_id: gfs + gfs2 support
authorKay Sievers <kay.sievers@suse.de>
Tue, 30 May 2006 14:01:28 +0000 (16:01 +0200)
committerKay Sievers <kay.sievers@suse.de>
Tue, 30 May 2006 14:01:28 +0000 (16:01 +0200)
Patch from Red Hat.

extras/volume_id/README
extras/volume_id/lib/Makefile
extras/volume_id/lib/exported_symbols
extras/volume_id/lib/gfs.c [new file with mode: 0644]
extras/volume_id/lib/libvolume_id.h
extras/volume_id/lib/volume_id.c

index be98df9b12c1390a2e2e557627210c2a51523e94..b458ca59baf631c7747eceb5820c7b4a351ef455 100644 (file)
@@ -36,6 +36,8 @@ squashfs               *      -      -     -
 minix                  *      -      -     *
 ocfs (1, 2)            *      *      *     *
 vxfs                   *      -      -     *
+nss (netware)          *             *     *
+gfs, gfs2              *      -      -     -
 
 Raid members are detected to prevent the probing for a filesystem
 -----------------------------------------------------------------
@@ -49,7 +51,8 @@ nvidia_raid            *      -      -     -
 promise_raid           *      -      -     -
 silicon_raid           *      -      -     -
 via_raid               *      -      -     -
-
+jmicron                *      -      -     -
+adaptec                *      -      -     -
 
 If no label is found vol_id exits with nonzero and the rule will be
 ignored.
index 8d8710d09add31d40ada4136585b583441dd5b92..90cd63e745e498e953ce1471d41d954c62a12f8c 100644 (file)
@@ -13,7 +13,7 @@ INSTALL_DATA  = ${INSTALL} -m 644
 INSTALL_LIB = ${INSTALL} -m 755
 
 SHLIB_CUR = 0
-SHLIB_REV = 65
+SHLIB_REV = 66
 SHLIB_AGE = 0
 SHLIB = libvolume_id.so.$(SHLIB_CUR).$(SHLIB_REV).$(SHLIB_AGE)
 
@@ -45,6 +45,7 @@ OBJS=                         \
        romfs.o                 \
        sysv.o                  \
        minix.o                 \
+       gfs.o                   \
        luks.o                  \
        ocfs.o                  \
        vxfs.o                  \
index df5f9f05c9da5c1f72f96d8ca157815f715fb781..3f10ec0c78d103e6f7509d415b7ac84680564ac7 100644 (file)
@@ -27,6 +27,8 @@
        volume_id_probe_xfs;
        volume_id_probe_squashfs;
        volume_id_probe_netware;
+       volume_id_probe_gfs;
+       volume_id_probe_gfs2;
 
        volume_id_probe_linux_raid;
        volume_id_probe_lvm1;
diff --git a/extras/volume_id/lib/gfs.c b/extras/volume_id/lib/gfs.c
new file mode 100644 (file)
index 0000000..40a4493
--- /dev/null
@@ -0,0 +1,120 @@
+/*
+ * volume_id - reads filesystem label and uuid
+ *
+ * Copyright (C) 2006 Red Hat, Inc. <redhat.com>
+ *
+ *     This program is free software; you can redistribute it and/or modify it
+ *     under the terms of the GNU General Public License as published by the
+ *     Free Software Foundation version 2 of the License.
+ */
+
+#ifndef _GNU_SOURCE
+#define _GNU_SOURCE 1
+#endif
+
+#ifdef HAVE_CONFIG_H
+#  include <config.h>
+#endif
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <unistd.h>
+#include <string.h>
+#include <errno.h>
+#include <ctype.h>
+
+#include "libvolume_id.h"
+#include "util.h"
+
+/* Common gfs/gfs2 constants: */
+#define GFS_MAGIC              0x01161970
+#define GFS_DEFAULT_BSIZE      4096
+#define GFS_SUPERBLOCK_OFFSET  (0x10 * GFS_DEFAULT_BSIZE)
+#define GFS_METATYPE_SB                1
+#define GFS_FORMAT_SB          100
+#define GFS_LOCKNAME_LEN       64
+
+/* gfs1 constants: */
+#define GFS_FORMAT_FS          1309
+#define GFS_FORMAT_MULTI       1401
+/* gfs2 constants: */
+#define GFS2_FORMAT_FS         1801
+#define GFS2_FORMAT_MULTI      1900
+
+struct gfs2_meta_header {
+       uint32_t mh_magic;
+       uint32_t mh_type;
+       uint64_t __pad0;          /* Was generation number in gfs1 */
+       uint32_t mh_format;
+       uint32_t __pad1;          /* Was incarnation number in gfs1 */
+};
+
+struct gfs2_inum {
+       uint64_t no_formal_ino;
+       uint64_t no_addr;
+};
+
+struct gfs2_sb {
+       struct gfs2_meta_header sb_header;
+
+       uint32_t sb_fs_format;
+       uint32_t sb_multihost_format;
+       uint32_t  __pad0;  /* Was superblock flags in gfs1 */
+
+       uint32_t sb_bsize;
+       uint32_t sb_bsize_shift;
+       uint32_t __pad1;   /* Was journal segment size in gfs1 */
+
+       struct gfs2_inum sb_master_dir; /* Was jindex dinode in gfs1 */
+       struct gfs2_inum __pad2; /* Was rindex dinode in gfs1 */
+       struct gfs2_inum sb_root_dir;
+
+       char sb_lockproto[GFS_LOCKNAME_LEN];
+       char sb_locktable[GFS_LOCKNAME_LEN];
+       /* In gfs1, quota and license dinodes followed */
+} PACKED;
+
+static int volume_id_probe_gfs_generic(struct volume_id *id, uint64_t off, int vers)
+{
+       struct gfs2_sb *sbd;
+
+       info("probing at offset 0x%llx", (unsigned long long) off);
+
+       sbd = (struct gfs2_sb *)
+               volume_id_get_buffer(id, off + GFS_SUPERBLOCK_OFFSET, sizeof(struct gfs2_sb));
+       if (sbd == NULL)
+               return -1;
+
+       if (be32_to_cpu(sbd->sb_header.mh_magic) == GFS_MAGIC &&
+               be32_to_cpu(sbd->sb_header.mh_type) == GFS_METATYPE_SB &&
+               be32_to_cpu(sbd->sb_header.mh_format) == GFS_FORMAT_SB) {
+               if (vers == 1) {
+                       if (be32_to_cpu(sbd->sb_fs_format) != GFS_FORMAT_FS ||
+                               be32_to_cpu(sbd->sb_multihost_format) != GFS_FORMAT_MULTI)
+                               return -1; /* not gfs1 */
+                       id->type = "gfs";
+               }
+               else if (vers == 2) {
+                       if (be32_to_cpu(sbd->sb_fs_format) != GFS2_FORMAT_FS ||
+                               be32_to_cpu(sbd->sb_multihost_format) != GFS2_FORMAT_MULTI)
+                               return -1; /* not gfs2 */
+                       id->type = "gfs2";
+               }
+               else
+                       return -1;
+               strcpy(id->type_version, "1");
+               volume_id_set_usage(id, VOLUME_ID_FILESYSTEM);
+               return 0;
+       }
+       return -1;
+}
+
+int volume_id_probe_gfs(struct volume_id *id, uint64_t off)
+{
+       return volume_id_probe_gfs_generic(id, off, 1);
+}
+
+int volume_id_probe_gfs2(struct volume_id *id, uint64_t off)
+{
+       return volume_id_probe_gfs_generic(id, off, 2);
+}
index df9fcd98910212ec0f90f1436b5aae738b45dc66..e34f4679e7503b30cb29cb8f2d7abb2e4dbf948e 100644 (file)
@@ -89,6 +89,8 @@ extern int volume_id_probe_vxfs(struct volume_id *id, uint64_t off);
 extern int volume_id_probe_xfs(struct volume_id *id, uint64_t off);
 extern int volume_id_probe_squashfs(struct volume_id *id, uint64_t off);
 extern int volume_id_probe_netware(struct volume_id *id, uint64_t off);
+extern int volume_id_probe_gfs(struct volume_id *id, uint64_t off);
+extern int volume_id_probe_gfs2(struct volume_id *id, uint64_t off);
 
 /* special formats */
 extern int volume_id_probe_linux_swap(struct volume_id *id, uint64_t off);
index fd95e7d3b0c557848e43c35d8227250109122e6b..bcdbe2feaaf1129a2b053ea1a9ae2a8d5408b2e8 100644 (file)
@@ -171,6 +171,12 @@ int volume_id_probe_filesystem(struct volume_id *id, uint64_t off, uint64_t size
        if (volume_id_probe_netware(id, off) == 0)
                goto found;
 
+       if (volume_id_probe_gfs(id, off) == 0)
+               goto found;
+
+       if (volume_id_probe_gfs2(id, off) == 0)
+               goto found;
+
        return -1;
 
 found: