Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sbin/mount_qemufwcfg Add virtual filesystem for QEMU Firmwar...



details:   https://anonhg.NetBSD.org/src/rev/e0f87fc7e9bc
branches:  trunk
changeset: 357759:e0f87fc7e9bc
user:      jmcneill <jmcneill%NetBSD.org@localhost>
date:      Sat Nov 25 23:23:39 2017 +0000

description:
Add virtual filesystem for QEMU Firmware Configuration interface.

diffstat:

 sbin/mount_qemufwcfg/Makefile  |   12 +
 sbin/mount_qemufwcfg/defs.h    |   88 +++++++++++++
 sbin/mount_qemufwcfg/fwcfg.c   |  259 +++++++++++++++++++++++++++++++++++++++++
 sbin/mount_qemufwcfg/virtdir.c |  226 +++++++++++++++++++++++++++++++++++
 sbin/mount_qemufwcfg/virtdir.h |   81 ++++++++++++
 5 files changed, 666 insertions(+), 0 deletions(-)

diffs (truncated from 686 to 300 lines):

diff -r 0a8f29be6396 -r e0f87fc7e9bc sbin/mount_qemufwcfg/Makefile
--- /dev/null   Thu Jan 01 00:00:00 1970 +0000
+++ b/sbin/mount_qemufwcfg/Makefile     Sat Nov 25 23:23:39 2017 +0000
@@ -0,0 +1,12 @@
+# $NetBSD: Makefile,v 1.1 2017/11/25 23:23:39 jmcneill Exp $
+
+PROG=  mount_qemufwcfg
+SRCS=  fwcfg.c virtdir.c
+DPADD+=        ${LIBREFUSE}
+LDADD= -lrefuse
+NOMAN= # defined
+WARNS= 3
+
+CPPFLAGS+=     -D_KERNTYPES
+
+.include <bsd.prog.mk>
diff -r 0a8f29be6396 -r e0f87fc7e9bc sbin/mount_qemufwcfg/defs.h
--- /dev/null   Thu Jan 01 00:00:00 1970 +0000
+++ b/sbin/mount_qemufwcfg/defs.h       Sat Nov 25 23:23:39 2017 +0000
@@ -0,0 +1,88 @@
+/* $NetBSD: defs.h,v 1.1 2017/11/25 23:23:39 jmcneill Exp $ */
+
+/*
+ * Copyright (c) 1999-2005 Alistair Crooks.  All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ * 3. The name of the author may not be used to endorse or promote
+ *    products derived from this software without specific prior written
+ *    permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
+ * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
+ * GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
+ * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+ * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
+ * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+#ifndef DEFS_H_
+#define DEFS_H_
+
+#include <sys/types.h>
+#include <sys/param.h>
+
+#ifdef HAVE_STDINT_H
+#include <stdint.h>
+#endif
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+#define NEWARRAY(type,ptr,size,where,action) do {                      \
+       if ((ptr = (type *) calloc(sizeof(type), (unsigned)(size))) == NULL) { \
+               (void) fprintf(stderr, "%s: can't allocate %lu bytes\n", \
+                       where, (unsigned long)(size * sizeof(type)));   \
+               action;                                                 \
+       }                                                               \
+} while( /* CONSTCOND */ 0)
+
+#define RENEW(type,ptr,size,where,action) do {                         \
+       type *_newptr;                                                  \
+       if ((_newptr = (type *) realloc(ptr, sizeof(type) * (size))) == NULL) { \
+               (void) fprintf(stderr, "%s: can't realloc %lu bytes\n", \
+                       where, (unsigned long)(size * sizeof(type)));   \
+               action;                                                 \
+       } else {                                                        \
+               ptr = _newptr;                                          \
+       }                                                               \
+} while( /* CONSTCOND */ 0)
+
+#define NEW(type, ptr, where, action)  NEWARRAY(type, ptr, 1, where, action)
+
+#define FREE(ptr)      (void) free(ptr)
+
+#define ALLOC(type, v, size, c, init, incr, where, action) do {                \
+       uint32_t        _newsize = size;                                \
+       if (size == 0) {                                                \
+               _newsize = init;                                        \
+               NEWARRAY(type, v, _newsize, where ": new", action);     \
+       } else if (c == size) {                                         \
+               _newsize = size + incr;                                 \
+               RENEW(type, v, _newsize, where ": renew", action);      \
+       }                                                               \
+       size = _newsize;                                                \
+} while( /* CONSTCOND */ 0)
+
+/*             (void) memset(&v[size], 0x0, sizeof(type) * incr);      \*/
+
+#define DEFINE_ARRAY(name, type)                                       \
+typedef struct name {                                                  \
+       uint32_t        c;                                              \
+       uint32_t        size;                                           \
+       type           *v;                                              \
+} name
+
+#endif /* !DEFS_H_ */
diff -r 0a8f29be6396 -r e0f87fc7e9bc sbin/mount_qemufwcfg/fwcfg.c
--- /dev/null   Thu Jan 01 00:00:00 1970 +0000
+++ b/sbin/mount_qemufwcfg/fwcfg.c      Sat Nov 25 23:23:39 2017 +0000
@@ -0,0 +1,259 @@
+/* $NetBSD: fwcfg.c,v 1.1 2017/11/25 23:23:39 jmcneill Exp $ */
+
+/*-
+ * Copyright (c) 2017 Jared McNeill <jmcneill%invisible.ca@localhost>
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
+ * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
+ * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
+ * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
+ * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
+ * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
+ * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+#include <sys/cdefs.h>
+__RCSID("$NetBSD: fwcfg.c,v 1.1 2017/11/25 23:23:39 jmcneill Exp $");
+
+#include <sys/ioctl.h>
+
+#include <err.h>
+#include <errno.h>
+#include <fcntl.h>
+#include <fuse.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <unistd.h>
+
+#include <dev/ic/qemufwcfgio.h>
+
+#include "virtdir.h"
+
+#define _PATH_FWCFG    "/dev/qemufwcfg"
+
+struct fwcfg_file {
+       uint32_t size;
+       uint16_t select;
+       uint16_t reserved;
+       char name[56];
+};
+
+static int fwcfg_fd;
+static mode_t fwcfg_dir_mask = 0555;
+static mode_t fwcfg_file_mask = 0444;
+static uid_t fwcfg_uid;
+static gid_t fwcfg_gid;
+
+static virtdir_t fwcfg_virtdir;
+
+static void
+set_index(uint16_t index)
+{
+       if (ioctl(fwcfg_fd, FWCFGIO_SET_INDEX, &index) != 0)
+               err(EXIT_FAILURE, "failed to set index 0x%04x", index);
+}
+
+static void
+read_data(void *buf, size_t buflen)
+{
+       if (read(fwcfg_fd, buf, buflen) != (ssize_t)buflen)
+               err(EXIT_FAILURE, "failed to read data");
+}
+
+static int
+fwcfg_getattr(const char *path, struct stat *st)
+{
+       virt_dirent_t *ep;
+
+       if (strcmp(path, "/") == 0) {
+               memset(st, 0, sizeof(*st));
+               st->st_mode = S_IFDIR | fwcfg_dir_mask;
+               st->st_nlink = 2;
+               return 0;
+       }
+
+       if ((ep = virtdir_find(&fwcfg_virtdir, path, strlen(path))) == NULL)
+               return -ENOENT;
+
+       switch (ep->type) {
+       case 'f':
+               memcpy(st, &fwcfg_virtdir.file, sizeof(*st));
+               st->st_size = ep->tgtlen;
+               st->st_mode = S_IFREG | fwcfg_file_mask;
+               break;
+       case 'd':
+               memcpy(st, &fwcfg_virtdir.dir, sizeof(*st));
+               st->st_mode = S_IFDIR | fwcfg_dir_mask;
+               break;
+       }
+       st->st_ino = virtdir_offset(&fwcfg_virtdir, ep) + 10;
+
+       return 0;
+}
+
+static int
+fwcfg_readdir(const char *path, void *buf, fuse_fill_dir_t filler,
+    off_t offset, struct fuse_file_info *fi)
+{
+       static VIRTDIR *dirp;
+       virt_dirent_t *dp;
+
+       if (offset == 0) {
+               if ((dirp = openvirtdir(&fwcfg_virtdir, path)) == NULL)
+                       return 0;
+               filler(buf, ".", NULL, 0);
+               filler(buf, "..", NULL, 0);
+       }
+       while ((dp = readvirtdir(dirp)) != NULL) {
+               if (filler(buf, dp->d_name, NULL, 0) != 0)
+                       return 0;
+       }
+       closevirtdir(dirp);
+       dirp = NULL;
+
+       return 0;
+}
+
+static int
+fwcfg_open(const char *path, struct fuse_file_info *fi)
+{
+       return 0;
+}
+
+static int
+fwcfg_read(const char *path, char *buf, size_t size, off_t offset,
+    struct fuse_file_info *fi)
+{
+       virt_dirent_t *ep;
+       uint8_t tmp[64];
+
+       if ((ep = virtdir_find(&fwcfg_virtdir, path, strlen(path))) == NULL)
+               return -ENOENT;
+
+       if (ep->select == 0)
+               return -ENOENT;
+
+       set_index(ep->select);
+
+       /* Seek to correct offset */
+       while (offset > 0) {
+               const int len = MIN(sizeof(tmp), (size_t)offset);
+               read_data(tmp, len);
+               offset -= len;
+       }
+
+       /* Read the data */
+       read_data(buf, size);
+
+       return size;
+}
+
+static int
+fwcfg_statfs(const char *path, struct statvfs *st)
+{
+       uint32_t count;
+
+       set_index(FW_CFG_FILE_DIR);
+       read_data(&count, sizeof(count));
+
+       memset(st, 0, sizeof(*st));
+       st->f_files = be32toh(count);
+
+       return 0;
+}
+
+static struct fuse_operations fwcfg_ops = {
+       .getattr = fwcfg_getattr,
+       .readdir = fwcfg_readdir,
+       .open = fwcfg_open,
+       .read = fwcfg_read,
+       .statfs = fwcfg_statfs,
+};
+
+static void



Home | Main Index | Thread Index | Old Index