155 lines
3.3 KiB
C
155 lines
3.3 KiB
C
/*
|
|
* Copyright (C) 2008 Karel Zak <kzak@redhat.com>
|
|
*
|
|
* This is written from sratch according to Linux kernel fs/sysv/super.c file.
|
|
* It seems that sysv probing code in libvolume_id and also in the original
|
|
* blkid is useless.
|
|
*
|
|
* This file may be redistributed under the terms of the
|
|
* GNU Lesser General Public License.
|
|
*/
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <stdint.h>
|
|
#include <stddef.h>
|
|
|
|
#include "superblocks.h"
|
|
|
|
#define XENIX_NICINOD 100
|
|
#define XENIX_NICFREE 100
|
|
|
|
struct xenix_super_block {
|
|
uint16_t s_isize;
|
|
uint32_t s_fsize;
|
|
uint16_t s_nfree;
|
|
uint32_t s_free[XENIX_NICFREE];
|
|
uint16_t s_ninode;
|
|
uint16_t s_inode[XENIX_NICINOD];
|
|
uint8_t s_flock;
|
|
uint8_t s_ilock;
|
|
uint8_t s_fmod;
|
|
uint8_t s_ronly;
|
|
uint32_t s_time;
|
|
uint32_t s_tfree;
|
|
uint16_t s_tinode;
|
|
uint16_t s_dinfo[4];
|
|
uint8_t s_fname[6];
|
|
uint8_t s_fpack[6];
|
|
uint8_t s_clean;
|
|
uint8_t s_fill[371];
|
|
uint32_t s_magic;
|
|
uint32_t s_type;
|
|
} __attribute__((packed));
|
|
|
|
|
|
#define SYSV_NICINOD 100
|
|
#define SYSV_NICFREE 50
|
|
|
|
struct sysv_super_block
|
|
{
|
|
uint16_t s_isize;
|
|
uint16_t s_pad0;
|
|
uint32_t s_fsize;
|
|
uint16_t s_nfree;
|
|
uint16_t s_pad1;
|
|
uint32_t s_free[SYSV_NICFREE];
|
|
uint16_t s_ninode;
|
|
uint16_t s_pad2;
|
|
uint16_t s_inode[SYSV_NICINOD];
|
|
uint8_t s_flock;
|
|
uint8_t s_ilock;
|
|
uint8_t s_fmod;
|
|
uint8_t s_ronly;
|
|
uint32_t s_time;
|
|
uint16_t s_dinfo[4];
|
|
uint32_t s_tfree;
|
|
uint16_t s_tinode;
|
|
uint16_t s_pad3;
|
|
uint8_t s_fname[6];
|
|
uint8_t s_fpack[6];
|
|
uint32_t s_fill[12];
|
|
uint32_t s_state;
|
|
uint32_t s_magic;
|
|
uint32_t s_type;
|
|
};
|
|
|
|
static int probe_xenix(blkid_probe pr, const struct blkid_idmag *mag)
|
|
{
|
|
struct xenix_super_block *sb;
|
|
|
|
sb = blkid_probe_get_sb(pr, mag, struct xenix_super_block);
|
|
if (!sb)
|
|
return -1;
|
|
blkid_probe_set_label(pr, sb->s_fname, sizeof(sb->s_fname));
|
|
return 0;
|
|
}
|
|
|
|
#define SYSV_BLOCK_SIZE 1024
|
|
|
|
/* Note that we don't probe for Coherent FS, this FS does not have
|
|
* magic string. (It requires to probe fname/fpack field..)
|
|
*/
|
|
static int probe_sysv(blkid_probe pr,
|
|
const struct blkid_idmag *mag __attribute__((__unused__)))
|
|
{
|
|
struct sysv_super_block *sb;
|
|
int blocks[] = {0, 9, 15, 18};
|
|
size_t i;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(blocks); i++) {
|
|
int off = blocks[i] * SYSV_BLOCK_SIZE + SYSV_BLOCK_SIZE/2;
|
|
|
|
sb = (struct sysv_super_block *)
|
|
blkid_probe_get_buffer(pr,
|
|
off,
|
|
sizeof(struct sysv_super_block));
|
|
if (!sb)
|
|
return -1;
|
|
|
|
if (sb->s_magic == cpu_to_le32(0xfd187e20) ||
|
|
sb->s_magic == cpu_to_be32(0xfd187e20)) {
|
|
|
|
if (blkid_probe_set_label(pr, sb->s_fname,
|
|
sizeof(sb->s_fname)))
|
|
return -1;
|
|
|
|
if (blkid_probe_set_magic(pr,
|
|
off + offsetof(struct sysv_super_block,
|
|
s_magic),
|
|
sizeof(sb->s_magic),
|
|
(unsigned char *) &sb->s_magic))
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
const struct blkid_idinfo xenix_idinfo =
|
|
{
|
|
.name = "xenix",
|
|
.usage = BLKID_USAGE_FILESYSTEM,
|
|
.probefunc = probe_xenix,
|
|
.magics =
|
|
{
|
|
{ .magic = "\x2b\x55\x44", .len = 3, .kboff = 1, .sboff = 0x400 },
|
|
{ .magic = "\x44\x55\x2b", .len = 3, .kboff = 1, .sboff = 0x400 },
|
|
{ NULL }
|
|
}
|
|
};
|
|
|
|
const struct blkid_idinfo sysv_idinfo =
|
|
{
|
|
.name = "sysv",
|
|
.usage = BLKID_USAGE_FILESYSTEM,
|
|
.probefunc = probe_sysv,
|
|
|
|
/* SYSV is BE and LE and superblock could be on four positions. It's
|
|
* simpler to probe for the magic string by .probefunc().
|
|
*/
|
|
.magics = BLKID_NONE_MAGIC
|
|
};
|
|
|