]> granicus.if.org Git - spl/commitdiff
Add XDR implementation
authorRicardo M. Correia <Ricardo.M.Correia@Sun.COM>
Tue, 10 Mar 2009 19:51:45 +0000 (19:51 +0000)
committerBrian Behlendorf <behlendorf1@llnl.gov>
Wed, 11 Mar 2009 20:00:26 +0000 (13:00 -0700)
Added proper XDR implementation (Lustre bug 17662), needed for on-disk
compatibility between platforms of different endianness.

include/rpc/types.h
include/rpc/xdr.h
module/spl/Makefile.in
module/spl/spl-xdr.c [new file with mode: 0644]

index 537d322f1e126a71a718a2dc1fbd82a294070878..e73fd9c0512f57f65e607f0a5fbfafdb43626f91 100644 (file)
@@ -1,6 +1,6 @@
 #ifndef _SPL_RPC_TYPES_H
 #define _SPL_RPC_TYPES_H
 
-#include <linux/sunrpc/types.h>
+typedef int bool_t;
 
 #endif /* SPL_RPC_TYPES_H */
index 3cd3574eb540dda1f35e7641dbf94b2e5eb4dda9..5b816546064830e82db76bccadc5c2a9fa348ad1 100644 (file)
@@ -1,6 +1,156 @@
+/*
+ *  This file is part of the SPL: Solaris Porting Layer.
+ *
+ *  Copyright (c) 2008 Sun Microsystems, Inc.
+ *
+ *  This 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; either version 2 of the License, or
+ *  (at your option) any later version.
+ *
+ *  This is distributed in the hope that it will be useful, but WITHOUT
+ *  ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+ *  FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
+ *  for more details.
+ *
+ *  You should have received a copy of the GNU General Public License along
+ *  with this program; if not, write to the Free Software Foundation, Inc.,
+ *  51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA.
+ */
+
 #ifndef _SPL_RPC_XDR_H
 #define _SPL_RPC_XDR_H
 
-#include <linux/sunrpc/xdr.h>
+#include <sys/types.h>
+#include <rpc/types.h>
+
+/*
+ * XDR enums and types.
+ */
+enum xdr_op {
+       XDR_ENCODE,
+       XDR_DECODE
+};
+
+struct xdr_ops;
+
+typedef struct {
+       struct xdr_ops *x_ops;      /* Also used to let caller know if
+                                      xdrmem_create() succeeds (sigh..) */
+       caddr_t         x_addr;     /* Current buffer addr */
+       caddr_t         x_addr_end; /* End of the buffer */
+       enum xdr_op     x_op;       /* Stream direction */
+} XDR;
+
+typedef bool_t (*xdrproc_t)(XDR *xdrs, void *ptr);
+
+struct xdr_ops {
+       bool_t (*xdr_control)(XDR *, int, void *);
+
+       bool_t (*xdr_char)(XDR *, char *);
+       bool_t (*xdr_u_short)(XDR *, unsigned short *);
+       bool_t (*xdr_u_int)(XDR *, unsigned *);
+       bool_t (*xdr_u_longlong_t)(XDR *, u_longlong_t *);
+
+       bool_t (*xdr_opaque)(XDR *, caddr_t, const uint_t);
+       bool_t (*xdr_string)(XDR *, char **, const uint_t);
+       bool_t (*xdr_array)(XDR *, caddr_t *, uint_t *, const uint_t,
+                           const uint_t, const xdrproc_t);
+};
+
+/*
+ * XDR control operator.
+ */
+#define XDR_GET_BYTES_AVAIL 1
+
+struct xdr_bytesrec {
+       bool_t xc_is_last_record;
+       size_t xc_num_avail;
+};
+
+typedef struct xdr_bytesrec xdr_bytesrec;
+
+/*
+ * XDR functions.
+ */
+void xdrmem_create(XDR *xdrs, const caddr_t addr, const uint_t size,
+    const enum xdr_op op);
+#define xdr_destroy(xdrs) ((void) 0) /* Currently not needed. If needed later,
+                                        we'll add it to struct xdr_ops */
+
+#define xdr_control(xdrs, req, info) (xdrs)->x_ops->xdr_control((xdrs),        \
+                                         (req), (info))
+
+/*
+ * For precaution, the following are defined as static inlines instead of macros
+ * to get some amount of type safety.
+ *
+ * Also, macros wouldn't work in the case where typecasting is done, because it
+ * must be possible to reference the functions' addresses by these names.
+ */
+static inline bool_t xdr_char(XDR *xdrs, char *cp)
+{
+       return xdrs->x_ops->xdr_char(xdrs, cp);
+}
+
+static inline bool_t xdr_u_short(XDR *xdrs, unsigned short *usp)
+{
+       return xdrs->x_ops->xdr_u_short(xdrs, usp);
+}
+
+static inline bool_t xdr_short(XDR *xdrs, short *sp)
+{
+       BUILD_BUG_ON(sizeof(short) != 2);
+       return xdrs->x_ops->xdr_u_short(xdrs, (unsigned short *) sp);
+}
+
+static inline bool_t xdr_u_int(XDR *xdrs, unsigned *up)
+{
+       return xdrs->x_ops->xdr_u_int(xdrs, up);
+}
+
+static inline bool_t xdr_int(XDR *xdrs, int *ip)
+{
+       BUILD_BUG_ON(sizeof(int) != 4);
+       return xdrs->x_ops->xdr_u_int(xdrs, (unsigned *) ip);
+}
+
+static inline bool_t xdr_u_longlong_t(XDR *xdrs, u_longlong_t *ullp)
+{
+       return xdrs->x_ops->xdr_u_longlong_t(xdrs, ullp);
+}
+
+static inline bool_t xdr_longlong_t(XDR *xdrs, longlong_t *llp)
+{
+       BUILD_BUG_ON(sizeof(longlong_t) != 8);
+       return xdrs->x_ops->xdr_u_longlong_t(xdrs, (u_longlong_t *) llp);
+}
+
+/*
+ * Fixed-length opaque data.
+ */
+static inline bool_t xdr_opaque(XDR *xdrs, caddr_t cp, const uint_t cnt)
+{
+       return xdrs->x_ops->xdr_opaque(xdrs, cp, cnt);
+}
+
+/*
+ * Variable-length string.
+ * The *sp buffer must have (maxsize + 1) bytes.
+ */
+static inline bool_t xdr_string(XDR *xdrs, char **sp, const uint_t maxsize)
+{
+       return xdrs->x_ops->xdr_string(xdrs, sp, maxsize);
+}
+
+/*
+ * Variable-length arrays.
+ */
+static inline bool_t xdr_array(XDR *xdrs, caddr_t *arrp, uint_t *sizep,
+    const uint_t maxsize, const uint_t elsize, const xdrproc_t elproc)
+{
+       return xdrs->x_ops->xdr_array(xdrs, arrp, sizep, maxsize, elsize,
+           elproc);
+}
 
 #endif /* SPL_RPC_XDR_H */
index 18f49434bdf41cbc1338743b044ec6407bce7c77..e9c8d347087f2a2bd527b73173eea86a4a89e5cc 100644 (file)
@@ -22,3 +22,4 @@ spl-objs += spl-atomic.o
 spl-objs += spl-mutex.o
 spl-objs += spl-kstat.o
 spl-objs += spl-condvar.o
+spl-objs += spl-xdr.o
diff --git a/module/spl/spl-xdr.c b/module/spl/spl-xdr.c
new file mode 100644 (file)
index 0000000..feaa59e
--- /dev/null
@@ -0,0 +1,517 @@
+/*
+ *  This file is part of the SPL: Solaris Porting Layer.
+ *
+ *  Copyright (c) 2008 Sun Microsystems, Inc.
+ *
+ *  This 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; either version 2 of the License, or
+ *  (at your option) any later version.
+ *
+ *  This is distributed in the hope that it will be useful, but WITHOUT
+ *  ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+ *  FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
+ *  for more details.
+ *
+ *  You should have received a copy of the GNU General Public License along
+ *  with this program; if not, write to the Free Software Foundation, Inc.,
+ *  51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA.
+ */
+
+#include <linux/string.h>
+
+#include <sys/kmem.h>
+#include <sys/debug.h>
+#include <sys/types.h>
+
+#include <rpc/types.h>
+#include <rpc/xdr.h>
+
+/*
+ * SPL's XDR mem implementation.
+ *
+ * This is used by libnvpair to serialize/deserialize the name-value pair data
+ * structures into byte arrays in a well-defined and portable manner.
+ *
+ * These data structures are used by the DMU/ZFS to flexibly manipulate various
+ * information in memory and later serialize it/deserialize it to disk.
+ * Examples of usages include the pool configuration, lists of pool and dataset
+ * properties, etc.
+ *
+ * Reference documentation for the XDR representation and XDR operations can be
+ * found in RFC 1832 and xdr(3), respectively.
+ *
+ * ===  Implementation shortcomings ===
+ *
+ * It is assumed that the following C types have the following sizes:
+ *
+ * char/unsigned char:      1 byte
+ * short/unsigned short:    2 bytes
+ * int/unsigned int:        4 bytes
+ * longlong_t/u_longlong_t: 8 bytes
+ *
+ * The C standard allows these types to be larger (and in the case of ints,
+ * shorter), so if that is the case on some compiler/architecture, the build
+ * will fail (on purpose).
+ *
+ * If someone wants to fix the code to work properly on such environments, then:
+ *
+ * 1) Preconditions should be added to xdrmem_enc functions to make sure the
+ *    caller doesn't pass arguments which exceed the expected range.
+ * 2) Functions which take signed integers should be changed to properly do
+ *    sign extension.
+ * 3) For ints with less than 32 bits, well.. I suspect you'll have bigger
+ *    problems than this implementation.
+ *
+ * It is also assumed that:
+ *
+ * 1) Chars have 8 bits.
+ * 2) We can always do 32-bit-aligned int memory accesses and byte-aligned
+ *    memcpy, memset and memcmp.
+ * 3) Arrays passed to xdr_array() are packed and the compiler/architecture
+ *    supports element-sized-aligned memory accesses.
+ * 4) Negative integers are natively stored in two's complement binary
+ *    representation.
+ *
+ * No checks are done for the 4 assumptions above, though.
+ *
+ * === Caller expectations ===
+ *
+ * Existing documentation does not describe the semantics of XDR operations very
+ * well.  Therefore, some assumptions about failure semantics will be made and
+ * will be described below:
+ *
+ * 1) If any encoding operation fails (e.g., due to lack of buffer space), the
+ * the stream should be considered valid only up to the encoding operation
+ * previous to the one that first failed. However, the stream size as returned
+ * by xdr_control() cannot be considered to be strictly correct (it may be
+ * bigger).
+ *
+ * Putting it another way, if there is an encoding failure it's undefined
+ * whether anything is added to the stream in that operation and therefore
+ * neither xdr_control() nor future encoding operations on the same stream can
+ * be relied upon to produce correct results.
+ *
+ * 2) If a decoding operation fails, it's undefined whether anything will be
+ * decoded into passed buffers/pointers during that operation, or what the
+ * values on those buffers will look like.
+ *
+ * Future decoding operations on the same stream will also have similar
+ * undefined behavior.
+ *
+ * 3) When the first decoding operation fails it is OK to trust the results of
+ * previous decoding operations on the same stream, as long as the caller
+ * expects a failure to be possible (e.g. due to end-of-stream).
+ *
+ * However, this is highly discouraged because the caller should know the
+ * stream size and should be coded to expect any decoding failure to be data
+ * corruption due to hardware, accidental or even malicious causes, which should
+ * be handled gracefully in all cases.
+ *
+ * In very rare situations where there are strong reasons to believe the data
+ * can be trusted to be valid and non-tampered with, then the caller may assume
+ * a decoding failure to be a bug (e.g. due to mismatched data types) and may
+ * fail non-gracefully.
+ *
+ * 4) Non-zero padding bytes will cause the decoding operation to fail.
+ *
+ * 5) Zero bytes on string types will also cause the decoding operation to fail.
+ *
+ * 6) It is assumed that either the pointer to the stream buffer given by the
+ * caller is 32-bit aligned or the architecture supports non-32-bit-aligned int
+ * memory accesses.
+ *
+ * 7) The stream buffer and encoding/decoding buffers/ptrs should not overlap.
+ *
+ * 8) If a caller passes pointers to non-kernel memory (e.g., pointers to user
+ * space or MMIO space), the computer may explode.
+ */
+
+static struct xdr_ops xdrmem_encode_ops;
+static struct xdr_ops xdrmem_decode_ops;
+
+void
+xdrmem_create(XDR *xdrs, const caddr_t addr, const uint_t size,
+    const enum xdr_op op)
+{
+       switch (op) {
+               case XDR_ENCODE:
+                       xdrs->x_ops = &xdrmem_encode_ops;
+                       break;
+               case XDR_DECODE:
+                       xdrs->x_ops = &xdrmem_decode_ops;
+                       break;
+               default:
+                       CWARN("Invalid op value: %d\n", op);
+                       xdrs->x_ops = NULL; /* Let the caller know we failed */
+                       return;
+       }
+
+       xdrs->x_op = op;
+       xdrs->x_addr = addr;
+       xdrs->x_addr_end = addr + size;
+
+       if (xdrs->x_addr_end < xdrs->x_addr) {
+               CWARN("Overflow while creating xdrmem: %p, %u\n", addr, size);
+               xdrs->x_ops = NULL;
+       }
+}
+EXPORT_SYMBOL(xdrmem_create);
+
+static bool_t
+xdrmem_control(XDR *xdrs, int req, void *info)
+{
+       struct xdr_bytesrec *rec = (struct xdr_bytesrec *) info;
+
+       if (req != XDR_GET_BYTES_AVAIL) {
+               CWARN("Called with unknown request: %d\n", req);
+               return FALSE;
+       }
+
+       rec->xc_is_last_record = TRUE; /* always TRUE in xdrmem streams */
+       rec->xc_num_avail = xdrs->x_addr_end - xdrs->x_addr;
+
+       return TRUE;
+}
+
+static bool_t
+xdrmem_enc_bytes(XDR *xdrs, caddr_t cp, const uint_t cnt)
+{
+       uint_t size = roundup(cnt, 4);
+       uint_t pad;
+
+       if (size < cnt)
+               return FALSE; /* Integer overflow */
+
+       if (xdrs->x_addr > xdrs->x_addr_end)
+               return FALSE;
+
+       if (xdrs->x_addr_end - xdrs->x_addr > size)
+               return FALSE;
+
+       memcpy(xdrs->x_addr, cp, cnt);
+
+       xdrs->x_addr += cnt;
+
+       pad = size - cnt;
+       if (pad > 0) {
+               memset(xdrs->x_addr, 0, pad);
+               xdrs->x_addr += pad;
+       }
+
+       return TRUE;
+}
+
+static bool_t
+xdrmem_dec_bytes(XDR *xdrs, caddr_t cp, const uint_t cnt)
+{
+       static uint32_t zero = 0;
+       uint_t size = roundup(cnt, 4);
+       uint_t pad;
+
+       if (size < cnt)
+               return FALSE; /* Integer overflow */
+
+       if (xdrs->x_addr > xdrs->x_addr_end)
+               return FALSE;
+
+       if (xdrs->x_addr_end - xdrs->x_addr > size)
+               return FALSE;
+
+       memcpy(cp, xdrs->x_addr, cnt);
+       xdrs->x_addr += cnt;
+
+       pad = size - cnt;
+       if (pad > 0) {
+               /* An inverted memchr() would be useful here... */
+               if (memcmp(&zero, xdrs->x_addr, pad) != 0)
+                       return FALSE;
+
+               xdrs->x_addr += pad;
+       }
+
+       return TRUE;
+}
+
+static bool_t
+xdrmem_enc_uint32(XDR *xdrs, uint32_t val)
+{
+       if (xdrs->x_addr + sizeof(uint32_t) > xdrs->x_addr_end)
+               return FALSE;
+
+       *((uint32_t *) xdrs->x_addr) = cpu_to_be32(val);
+
+       xdrs->x_addr += sizeof(uint32_t);
+
+       return TRUE;
+}
+
+static bool_t
+xdrmem_dec_uint32(XDR *xdrs, uint32_t *val)
+{
+       if (xdrs->x_addr + sizeof(uint32_t) > xdrs->x_addr_end)
+               return FALSE;
+
+       *val = be32_to_cpu(*((uint32_t *) xdrs->x_addr));
+
+       xdrs->x_addr += sizeof(uint32_t);
+
+       return TRUE;
+}
+
+static bool_t
+xdrmem_enc_char(XDR *xdrs, char *cp)
+{
+       uint32_t val;
+       
+       BUILD_BUG_ON(sizeof(char) != 1);
+       val = *((unsigned char *) cp);
+
+       return xdrmem_enc_uint32(xdrs, val);
+}
+
+static bool_t
+xdrmem_dec_char(XDR *xdrs, char *cp)
+{
+       uint32_t val;
+
+       BUILD_BUG_ON(sizeof(char) != 1);
+
+       if (!xdrmem_dec_uint32(xdrs, &val))
+               return FALSE;
+
+       /*
+        * If any of the 3 other bytes are non-zero then val will be greater
+        * than 0xff and we fail because according to the RFC, this block does
+        * not have a char encoded in it.
+        */
+       if (val > 0xff)
+               return FALSE;
+
+       *((unsigned char *) cp) = val;
+
+       return TRUE;
+}
+
+static bool_t
+xdrmem_enc_ushort(XDR *xdrs, unsigned short *usp)
+{
+       BUILD_BUG_ON(sizeof(unsigned short) != 2);
+
+       return xdrmem_enc_uint32(xdrs, *usp);
+}
+
+static bool_t
+xdrmem_dec_ushort(XDR *xdrs, unsigned short *usp)
+{
+       uint32_t val;
+
+       BUILD_BUG_ON(sizeof(unsigned short) != 2);
+
+       if (!xdrmem_dec_uint32(xdrs, &val))
+               return FALSE;
+
+       /*
+        * Short ints are not in the RFC, but we assume similar logic as in
+        * xdrmem_dec_char().
+        */
+       if (val > 0xffff)
+               return FALSE;
+
+       *usp = val;
+
+       return TRUE;
+}
+
+static bool_t
+xdrmem_enc_uint(XDR *xdrs, unsigned *up)
+{
+       BUILD_BUG_ON(sizeof(unsigned) != 4);
+
+       return xdrmem_enc_uint32(xdrs, *up);
+}
+
+static bool_t
+xdrmem_dec_uint(XDR *xdrs, unsigned *up)
+{
+       BUILD_BUG_ON(sizeof(unsigned) != 4);
+
+       return xdrmem_dec_uint32(xdrs, (uint32_t *) up);
+}
+
+static bool_t
+xdrmem_enc_ulonglong(XDR *xdrs, u_longlong_t *ullp)
+{
+       BUILD_BUG_ON(sizeof(u_longlong_t) != 8);
+
+       if (!xdrmem_enc_uint32(xdrs, *ullp >> 32))
+               return FALSE;
+
+       return xdrmem_enc_uint32(xdrs, *ullp & 0xffffffff);
+}
+
+static bool_t
+xdrmem_dec_ulonglong(XDR *xdrs, u_longlong_t *ullp)
+{
+       uint32_t low, high;
+
+       BUILD_BUG_ON(sizeof(u_longlong_t) != 8);
+
+       if (!xdrmem_dec_uint32(xdrs, &high))
+               return FALSE;
+       if (!xdrmem_dec_uint32(xdrs, &low))
+               return FALSE;
+
+       *ullp = ((u_longlong_t) high << 32) | low;
+
+       return TRUE;
+}
+
+static bool_t
+xdr_enc_array(XDR *xdrs, caddr_t *arrp, uint_t *sizep, const uint_t maxsize,
+    const uint_t elsize, const xdrproc_t elproc)
+{
+       uint_t i;
+       caddr_t addr = *arrp;
+
+       if (*sizep > maxsize || *sizep > UINT_MAX / elsize)
+               return FALSE;
+
+       if (!xdrmem_enc_uint(xdrs, sizep))
+               return FALSE;
+
+       for (i = 0; i < *sizep; i++) {
+               if (!elproc(xdrs, addr))
+                       return FALSE;
+               addr += elsize;
+       }
+
+       return TRUE;
+}
+
+static bool_t
+xdr_dec_array(XDR *xdrs, caddr_t *arrp, uint_t *sizep, const uint_t maxsize,
+    const uint_t elsize, const xdrproc_t elproc)
+{
+       uint_t i, size;
+       bool_t alloc = FALSE;
+       caddr_t addr;
+
+       if (!xdrmem_dec_uint(xdrs, sizep))
+               return FALSE;
+
+       size = *sizep;
+
+       if (size > maxsize || size > UINT_MAX / elsize)
+               return FALSE;
+
+       /*
+        * The Solaris man page says: "If *arrp is NULL when decoding,
+        * xdr_array() allocates memory and *arrp points to it".
+        */
+       if (*arrp == NULL) {
+               BUILD_BUG_ON(sizeof(uint_t) > sizeof(size_t));
+
+               *arrp = kmem_alloc(size * elsize, KM_NOSLEEP);
+               if (*arrp == NULL)
+                       return FALSE;
+
+               alloc = TRUE;
+       }
+
+       addr = *arrp;
+
+       for (i = 0; i < size; i++) {
+               if (!elproc(xdrs, addr)) {
+                       if (alloc)
+                               kmem_free(*arrp, size * elsize);
+                       return FALSE;
+               }
+               addr += elsize;
+       }
+
+       return TRUE;
+}
+
+static bool_t
+xdr_enc_string(XDR *xdrs, char **sp, const uint_t maxsize)
+{
+       size_t slen = strlen(*sp);
+       uint_t len;
+
+       if (slen > maxsize)
+               return FALSE;
+
+       len = slen;
+
+       if (!xdrmem_enc_uint(xdrs, &len))
+               return FALSE;
+
+       return xdrmem_enc_bytes(xdrs, *sp, len);
+}
+
+static bool_t
+xdr_dec_string(XDR *xdrs, char **sp, const uint_t maxsize)
+{
+       uint_t size;
+       bool_t alloc = FALSE;
+
+       if (!xdrmem_dec_uint(xdrs, &size))
+               return FALSE;
+
+       if (size > maxsize || size > UINT_MAX - 1)
+               return FALSE;
+
+       /*
+        * Solaris man page: "If *sp is NULL when decoding, xdr_string()
+        * allocates memory and *sp points to it".
+        */
+       if (*sp == NULL) {
+               BUILD_BUG_ON(sizeof(uint_t) > sizeof(size_t));
+
+               *sp = kmem_alloc(size + 1, KM_NOSLEEP);
+               if (*sp == NULL)
+                       return FALSE;
+
+               alloc = TRUE;
+       }
+
+       if (!xdrmem_dec_bytes(xdrs, *sp, size))
+               goto fail;
+
+       if (memchr(*sp, 0, size) != NULL)
+               goto fail;
+
+       (*sp)[size] = '\0';
+
+       return TRUE;
+
+fail:
+       if (alloc)
+               kmem_free(*sp, size + 1);
+
+       return FALSE;
+}
+
+static struct xdr_ops xdrmem_encode_ops = {
+       .xdr_control      = xdrmem_control,
+       .xdr_char         = xdrmem_enc_char,
+       .xdr_u_short      = xdrmem_enc_ushort,
+       .xdr_u_int        = xdrmem_enc_uint,
+       .xdr_u_longlong_t = xdrmem_enc_ulonglong,
+       .xdr_opaque       = xdrmem_enc_bytes,
+       .xdr_string       = xdr_enc_string,
+       .xdr_array        = xdr_enc_array
+};
+
+static struct xdr_ops xdrmem_decode_ops = {
+       .xdr_control      = xdrmem_control,
+       .xdr_char         = xdrmem_dec_char,
+       .xdr_u_short      = xdrmem_dec_ushort,
+       .xdr_u_int        = xdrmem_dec_uint,
+       .xdr_u_longlong_t = xdrmem_dec_ulonglong,
+       .xdr_opaque       = xdrmem_dec_bytes,
+       .xdr_string       = xdr_dec_string,
+       .xdr_array        = xdr_dec_array
+};
+