[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
[PATCH 3/4] Add ZFS envblock functions
From: |
Paul Dagnelie |
Subject: |
[PATCH 3/4] Add ZFS envblock functions |
Date: |
Tue, 25 Feb 2020 11:26:45 -0800 |
This patch adds a ZFS implementation of the new envblock functions,
storing the data for the envblock in the second padding area of the
label. This data is protected by an embedded checksum and is stored
redundantly, so even though it is not part of the block tree it
provides reasonable reliability.
commit 0f108fee27917afd72d834620db8f8b1e7ca1699
Author: Paul Dagnelie <address@hidden>
AuthorDate: Mon Feb 24 14:29:35 2020 -0800
Commit: Paul Dagnelie <address@hidden>
CommitDate: Tue Feb 25 10:08:08 2020 -0800
Add ZFS envblock functions
---
grub-core/fs/zfs/zfs.c | 129 +++++++++++++++++++++++++++++++++++++++----
include/grub/zfs/vdev_impl.h | 33 ++++++-----
2 files changed, 134 insertions(+), 28 deletions(-)
diff --git a/grub-core/fs/zfs/zfs.c b/grub-core/fs/zfs/zfs.c
index 2f72e42bf..1ee9dd56b 100644
--- a/grub-core/fs/zfs/zfs.c
+++ b/grub-core/fs/zfs/zfs.c
@@ -30,6 +30,7 @@
*
*/
+#include <stddef.h>
#include <grub/err.h>
#include <grub/file.h>
#include <grub/mm.h>
@@ -1146,7 +1147,6 @@ scan_disk (grub_device_t dev, struct grub_zfs_data *data,
{
int label = 0;
uberblock_phys_t *ub_array, *ubbest = NULL;
- vdev_boot_header_t *bh;
grub_err_t err;
int vdevnum;
struct grub_zfs_device_desc desc;
@@ -1155,13 +1155,6 @@ scan_disk (grub_device_t dev, struct grub_zfs_data *data,
if (!ub_array)
return grub_errno;
- bh = grub_malloc (VDEV_BOOT_HEADER_SIZE);
- if (!bh)
- {
- grub_free (ub_array);
- return grub_errno;
- }
-
vdevnum = VDEV_LABELS;
desc.dev = dev;
@@ -1175,7 +1168,7 @@ scan_disk (grub_device_t dev, struct grub_zfs_data *data,
{
desc.vdev_phys_sector
= label * (sizeof (vdev_label_t) >> SPA_MINBLOCKSHIFT)
- + ((VDEV_SKIP_SIZE + VDEV_BOOT_HEADER_SIZE) >> SPA_MINBLOCKSHIFT)
+ + ((VDEV_PAD_SIZE * 2) >> SPA_MINBLOCKSHIFT)
+ (label < VDEV_LABELS / 2 ? 0 :
ALIGN_DOWN (grub_disk_get_size (dev->disk), sizeof (vdev_label_t))
- VDEV_LABELS * (sizeof (vdev_label_t) >> SPA_MINBLOCKSHIFT));
@@ -1184,6 +1177,7 @@ scan_disk (grub_device_t dev, struct grub_zfs_data *data,
err = grub_disk_read (dev->disk, desc.vdev_phys_sector
+ (VDEV_PHYS_SIZE >> SPA_MINBLOCKSHIFT),
0, VDEV_UBERBLOCK_RING, (char *) ub_array);
+ if (label == 0)
if (err)
{
grub_errno = GRUB_ERR_NONE;
@@ -1219,12 +1213,10 @@ scan_disk (grub_device_t dev, struct
grub_zfs_data *data,
continue;
#endif
grub_free (ub_array);
- grub_free (bh);
return GRUB_ERR_NONE;
}
grub_free (ub_array);
- grub_free (bh);
return grub_error (GRUB_ERR_BAD_FS, "couldn't find a valid label");
}
@@ -3765,6 +3757,58 @@ zfs_mtime (grub_device_t device, grub_int32_t *mt)
return GRUB_ERR_NONE;
}
+static grub_err_t
+zfs_devs_read_zbt (struct grub_zfs_data *data, grub_uint64_t offset,
char *buf, grub_size_t len)
+{
+ grub_err_t err = GRUB_ERR_NONE;
+ zio_cksum_t zc;
+ unsigned int i;
+ ZIO_SET_CHECKSUM(&zc, offset, 0, 0, 0);
+
+ for (i = 0; i < data->n_devices_attached; i++)
+ {
+ err = grub_disk_read (data->devices_attached[i].dev->disk,
+ offset >> SPA_MINBLOCKSHIFT,
+ offset & ((1 << SPA_MINBLOCKSHIFT) - 1),
+ len, buf);
+ if (err)
+ continue;
+ ZIO_SET_CHECKSUM(&zc, offset, 0, 0, 0);
+ err = zio_checksum_verify (zc, ZIO_CHECKSUM_LABEL,
GRUB_ZFS_LITTLE_ENDIAN,
+ buf, len);
+ if (!err)
+ return GRUB_ERR_NONE;
+ }
+ return err;
+}
+
+static grub_err_t
+grub_zfs_envblk_open (struct grub_file *file)
+{
+ grub_err_t err;
+ struct grub_zfs_data *data;
+ vdev_boot_envblock_t *vbe;
+ int l;
+ file->offset = 0;
+ data = zfs_mount (file->device);
+ file->data = data;
+ data->file_buf = grub_malloc (sizeof (vdev_boot_envblock_t));
+ for (l = 0; l < VDEV_LABELS / 2; l++)
+ {
+ grub_uint64_t offset = l * sizeof (vdev_label_t) + offsetof
(vdev_label_t, vl_be);
+ err = zfs_devs_read_zbt (data, offset, data->file_buf,
+ sizeof (vdev_boot_envblock_t));
+ if (err == GRUB_ERR_NONE)
+ {
+ vbe = (vdev_boot_envblock_t *)data->file_buf;
+ file->size = grub_strlen (vbe->vbe_bootenv);
+ return err;
+ }
+ }
+ zfs_unmount (data);
+ return err;
+}
+
/*
* zfs_open() locates a file in the rootpool by following the
* MOS and places the dnode of the file in the memory address DNODE.
@@ -3850,6 +3894,19 @@ grub_zfs_open (struct grub_file *file, const
char *fsfilename)
return GRUB_ERR_NONE;
}
+static grub_ssize_t
+grub_zfs_envblk_read (grub_file_t file, char *buf, grub_size_t len)
+{
+ struct grub_zfs_data *data = (struct grub_zfs_data *) file->data;
+ grub_ssize_t olen = len;
+ grub_uint64_t offset = file->offset + offsetof
(vdev_boot_envblock_t, vbe_bootenv);
+
+ if (len + file->offset > file->size)
+ olen = file->size - file->offset;
+ grub_memmove (buf, data->file_buf + offset, olen);
+ return olen;
+}
+
static grub_ssize_t
grub_zfs_read (grub_file_t file, char *buf, grub_size_t len)
{
@@ -3859,6 +3916,9 @@ grub_zfs_read (grub_file_t file, char *buf,
grub_size_t len)
grub_size_t read;
grub_err_t err;
+ if (grub_file_envblk (file))
+ return grub_zfs_envblk_read (file, buf, len);
+
/*
* If offset is in memory, move it into the buffer provided and return.
*/
@@ -3924,6 +3984,49 @@ grub_zfs_read (grub_file_t file, char *buf,
grub_size_t len)
return len;
}
+static grub_err_t
+grub_zfs_envblk_write (struct grub_file *file, char *buf, grub_size_t len)
+{
+ grub_uint64_t offset = file->offset + offsetof
(vdev_boot_envblock_t, vbe_bootenv);
+ grub_err_t err = GRUB_ERR_NONE;
+ struct grub_zfs_data *data = file->data;
+ unsigned int i, l;
+ zio_cksum_t cksum;
+ vdev_boot_envblock_t *vbe = (vdev_boot_envblock_t *)data->file_buf;
+ if (len + file->offset > file->size)
+ return GRUB_ERR_OUT_OF_RANGE;
+
+ grub_memmove (data->file_buf + offset, buf, len);
+
+ for (l = 0; l < VDEV_LABELS / 2; l++)
+ {
+ offset = l * sizeof (vdev_label_t) + offsetof (vdev_label_t, vl_be);
+ vbe->vbe_zbt.zec_magic = ZEC_MAGIC;
+ ZIO_SET_CHECKSUM(&vbe->vbe_zbt.zec_cksum, offset, 0, 0, 0);
+ vbe->vbe_zbt.zec_magic = ZEC_MAGIC;
+ zio_checksum_SHA256(vbe, VDEV_PAD_SIZE, GRUB_ZFS_LITTLE_ENDIAN, &cksum);
+ vbe->vbe_zbt.zec_cksum = cksum;
+
+ for (i = 0; i < data->n_devices_attached; i++)
+ {
+ struct grub_zfs_device_desc *desc = &data->devices_attached[i];
+ int num_sectors = VDEV_PAD_SIZE >> desc->ashift;
+ int label_seek = l * sizeof (vdev_label_t) >> desc->ashift;
+ int j;
+ for (j = 0; j < num_sectors; j++)
+ {
+ grub_err_t err2;
+ err2 = grub_disk_write (desc->dev->disk, label_seek +
num_sectors + j, 0,
+ 1 << desc->ashift,
+ data->file_buf + (j << desc->ashift));
+ if (err2 != GRUB_ERR_NONE)
+ err = err2;
+ }
+ }
+ }
+ return err;
+}
+
static grub_err_t
grub_zfs_close (grub_file_t file)
{
@@ -4360,6 +4463,10 @@ static struct grub_fs grub_zfs_fs = {
.reserved_first_sector = 1,
.blocklist_install = 0,
#endif
+ .fs_envblk_open = grub_zfs_envblk_open,
+ .fs_envblk_read = grub_zfs_read,
+ .fs_envblk_write = grub_zfs_envblk_write,
+ .fs_envblk_close = grub_zfs_close,
.next = 0
};
diff --git a/include/grub/zfs/vdev_impl.h b/include/grub/zfs/vdev_impl.h
index 9b5f0a7a8..48ec59003 100644
--- a/include/grub/zfs/vdev_impl.h
+++ b/include/grub/zfs/vdev_impl.h
@@ -23,34 +23,33 @@
#ifndef _SYS_VDEV_IMPL_H
#define _SYS_VDEV_IMPL_H
-#define VDEV_SKIP_SIZE (8 << 10)
-#define VDEV_BOOT_HEADER_SIZE (8 << 10)
+#define VDEV_PAD_SIZE (8 << 10)
#define VDEV_PHYS_SIZE (112 << 10)
#define VDEV_UBERBLOCK_RING (128 << 10)
-/* ZFS boot block */
-#define VDEV_BOOT_MAGIC 0x2f5b007b10cULL
-#define VDEV_BOOT_VERSION 1 /* version number */
-
-typedef struct vdev_boot_header {
- grub_uint64_t vb_magic; /* VDEV_BOOT_MAGIC */
- grub_uint64_t vb_version; /* VDEV_BOOT_VERSION */
- grub_uint64_t vb_offset; /* start offset (bytes) */
- grub_uint64_t vb_size; /* size (bytes) */
- char vb_pad[VDEV_BOOT_HEADER_SIZE - 4 * sizeof (grub_uint64_t)];
-} vdev_boot_header_t;
-
typedef struct vdev_phys {
char vp_nvlist[VDEV_PHYS_SIZE - sizeof (zio_eck_t)];
zio_eck_t vp_zbt;
} vdev_phys_t;
+typedef enum vbe_vers {
+ VB_RAW,
+ VB_NVLIST
+} vb_evers_t;
+
+typedef struct vdev_boot_envblock {
+ grub_uint64_t vbe_version;
+ char vbe_bootenv[VDEV_PAD_SIZE - sizeof (grub_uint64_t) -
+ sizeof (zio_eck_t)];
+ zio_eck_t vbe_zbt;
+} vdev_boot_envblock_t;
+
typedef struct vdev_label {
- char vl_pad[VDEV_SKIP_SIZE]; /* 8K */
- vdev_boot_header_t vl_boot_header; /* 8K */
+ char vl_pad1[VDEV_PAD_SIZE]; /* 8K */
+ vdev_boot_envblock_t vl_be; /* 8K */
vdev_phys_t vl_vdev_phys; /* 112K */
char vl_uberblock[VDEV_UBERBLOCK_RING]; /* 128K */
-} vdev_label_t; /* 256K total */
+} vdev_label_t; /* 256K total */
/*
* Size and offset of embedded boot loader region on each label.
--
Paul Dagnelie
[Prev in Thread] |
Current Thread |
[Next in Thread] |
- [PATCH 3/4] Add ZFS envblock functions,
Paul Dagnelie <=