1 /*
   2  * CDDL HEADER START
   3  *
   4  * The contents of this file are subject to the terms of the
   5  * Common Development and Distribution License (the "License").
   6  * You may not use this file except in compliance with the License.
   7  *
   8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
   9  * or http://www.opensolaris.org/os/licensing.
  10  * See the License for the specific language governing permissions
  11  * and limitations under the License.
  12  *
  13  * When distributing Covered Code, include this CDDL HEADER in each
  14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
  15  * If applicable, add the following below this CDDL HEADER, with the
  16  * fields enclosed by brackets "[]" replaced with your own identifying
  17  * information: Portions Copyright [yyyy] [name of copyright owner]
  18  *
  19  * CDDL HEADER END
  20  */
  21 /*
  22  * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
  23  * Use is subject to license terms.
  24  *
  25  * Copyright 2017 RackTop Systems.
  26  */
  27 
  28 #include <rpc/types.h>
  29 #include <rpc/xdr.h>
  30 #include <sys/types.h>
  31 #include <sys/sdt.h>
  32 #include <rpc/auth.h>
  33 #include <rpc/rpc_rdma.h>
  34 
  35 struct private {
  36         int     min_chunk;
  37         uint_t  flags;                  /* controls setting for rdma xdr */
  38         int     num_chunk;
  39         caddr_t inline_buf;             /* temporary buffer for xdr inlining */
  40         int     inline_len;             /* inline buffer length */
  41         uint_t  xp_reply_chunk_len;
  42         uint_t  xp_reply_chunk_len_alt;
  43 };
  44 
  45 /* ARGSUSED */
  46 static bool_t
  47 x_putint32_t(XDR *xdrs, int32_t *ip)
  48 {
  49         xdrs->x_handy += BYTES_PER_XDR_UNIT;
  50         return (TRUE);
  51 }
  52 
  53 /* ARGSUSED */
  54 static bool_t
  55 x_putbytes(XDR *xdrs, char *bp, int len)
  56 {
  57         struct private *xdrp = (struct private *)xdrs->x_private;
  58 
  59         /*
  60          * min_chunk = 0, means that the stream of bytes, to estimate size of,
  61          * contains no chunks to seperate out. See xdrrdma_putbytes()
  62          */
  63         if (len < xdrp->min_chunk || !(xdrp->flags & XDR_RDMA_CHUNK)) {
  64                 xdrs->x_handy += len;
  65                 return (TRUE);
  66         }
  67         /*
  68          * Chunk item. No impact on xdr size.
  69          */
  70         xdrp->num_chunk++;
  71 
  72         return (TRUE);
  73 }
  74 
  75 static uint_t
  76 x_getpostn(XDR *xdrs)
  77 {
  78         return (xdrs->x_handy);
  79 }
  80 
  81 /* ARGSUSED */
  82 static bool_t
  83 x_setpostn(XDR *xdrs, uint_t pos)
  84 {
  85         /* This is not allowed */
  86         return (FALSE);
  87 }
  88 
  89 /* ARGSUSED */
  90 static bool_t
  91 x_control(XDR *xdrs, int request, void *info)
  92 {
  93         int32_t *int32p;
  94         uint_t in_flags;
  95         rdma_chunkinfo_t *rcip = NULL;
  96         rdma_chunkinfo_lengths_t *rcilp = NULL;
  97         struct private *xdrp = (struct private *)xdrs->x_private;
  98 
  99         switch (request) {
 100         case XDR_RDMA_SET_FLAGS:
 101                 /*
 102                  * Set the flags provided in the *info in xp_flags for rdma xdr
 103                  * stream control.
 104                  */
 105                 int32p = (int32_t *)info;
 106                 in_flags = (uint_t)(*int32p);
 107 
 108                 xdrp->flags = in_flags;
 109                 return (TRUE);
 110 
 111         case XDR_RDMA_GET_FLAGS:
 112                 /*
 113                  * Get the flags provided in xp_flags return through *info
 114                  */
 115                 int32p = (int32_t *)info;
 116 
 117                 *int32p = (int32_t)xdrp->flags;
 118                 return (TRUE);
 119 
 120         case XDR_RDMA_GET_CHUNK_LEN:
 121                 rcilp = (rdma_chunkinfo_lengths_t *)info;
 122                 rcilp->rcil_len = xdrp->xp_reply_chunk_len;
 123                 rcilp->rcil_len_alt = xdrp->xp_reply_chunk_len_alt;
 124 
 125                 return (TRUE);
 126 
 127         case XDR_RDMA_ADD_CHUNK:
 128                 rcip = (rdma_chunkinfo_t *)info;
 129 
 130                 switch (rcip->rci_type) {
 131                 case RCI_WRITE_UIO_CHUNK:
 132                         xdrp->xp_reply_chunk_len_alt += rcip->rci_len;
 133                         break;
 134 
 135                 case RCI_WRITE_ADDR_CHUNK:
 136                         xdrp->xp_reply_chunk_len_alt += rcip->rci_len;
 137                         break;
 138 
 139                 case RCI_REPLY_CHUNK:
 140                         xdrp->xp_reply_chunk_len += rcip->rci_len;
 141                         break;
 142                 }
 143                 return (TRUE);
 144 
 145         default:
 146                 return (FALSE);
 147         }
 148 }
 149 
 150 /* ARGSUSED */
 151 static rpc_inline_t *
 152 x_inline(XDR *xdrs, int len)
 153 {
 154         struct private *xdrp = (struct private *)xdrs->x_private;
 155 
 156         if (len == 0) {
 157                 return (NULL);
 158         }
 159         if (xdrs->x_op != XDR_ENCODE) {
 160                 return (NULL);
 161         }
 162         if (len >= xdrp->min_chunk) {
 163                 return (NULL);
 164         }
 165         if (len <= xdrp->inline_len) {
 166                 /* inline_buf was already allocated, just reuse it */
 167                 xdrs->x_handy += len;
 168                 return ((rpc_inline_t *)xdrp->inline_buf);
 169         } else {
 170                 /* Free the earlier space and allocate new area */
 171                 if (xdrp->inline_buf)
 172                         mem_free(xdrp->inline_buf, xdrp->inline_len);
 173                 if ((xdrp->inline_buf = (caddr_t)mem_alloc(len)) == NULL) {
 174                         xdrp->inline_len = 0;
 175                         return (NULL);
 176                 }
 177                 xdrp->inline_len = len;
 178                 xdrs->x_handy += len;
 179                 return ((rpc_inline_t *)xdrp->inline_buf);
 180         }
 181 }
 182 
 183 static int
 184 harmless()
 185 {
 186         /* Always return FALSE/NULL, as the case may be */
 187         return (0);
 188 }
 189 
 190 static void
 191 x_destroy(XDR *xdrs)
 192 {
 193         struct private *xdrp = (struct private *)xdrs->x_private;
 194 
 195         xdrs->x_handy = 0;
 196         if (xdrp) {
 197                 if (xdrp->inline_buf)
 198                         mem_free(xdrp->inline_buf, xdrp->inline_len);
 199                 mem_free(xdrp, sizeof (struct private));
 200                 xdrs->x_private = NULL;
 201         }
 202         xdrs->x_base = 0;
 203 }
 204 
 205 static bool_t
 206 xdrrdma_common(XDR *xdrs, int min_chunk)
 207 {
 208         struct private *xdrp;
 209 
 210         xdrs->x_ops = xdrrdma_xops();
 211         xdrs->x_op = XDR_ENCODE;
 212         xdrs->x_handy = 0;
 213         xdrs->x_base = NULL;
 214         xdrs->x_private = kmem_zalloc(sizeof (struct private), KM_SLEEP);
 215         xdrp = (struct private *)xdrs->x_private;
 216         xdrp->min_chunk = min_chunk;
 217         xdrp->flags = 0;
 218         if (xdrp->min_chunk != 0)
 219                 xdrp->flags |= XDR_RDMA_CHUNK;
 220 
 221         xdrp->xp_reply_chunk_len = 0;
 222         xdrp->xp_reply_chunk_len_alt = 0;
 223 
 224         return (TRUE);
 225 }
 226 
 227 unsigned int
 228 xdrrdma_sizeof(xdrproc_t func, void *data, int min_chunk,
 229     uint_t *reply_size, uint_t *reply_size_alt)
 230 {
 231         XDR x;
 232         struct xdr_ops ops;
 233         bool_t stat;
 234         struct private *xdrp;
 235 
 236         x.x_ops = &ops;
 237         (void) xdrrdma_common(&x, min_chunk);
 238 
 239         stat = func(&x, data);
 240         xdrp = (struct private *)x.x_private;
 241         if (xdrp) {
 242                 if (reply_size != NULL)
 243                         *reply_size = xdrp->xp_reply_chunk_len;
 244                 if (reply_size_alt != NULL)
 245                         *reply_size_alt = xdrp->xp_reply_chunk_len_alt;
 246                 if (xdrp->inline_buf)
 247                         mem_free(xdrp->inline_buf, xdrp->inline_len);
 248                 mem_free(xdrp, sizeof (struct private));
 249         }
 250         return (stat == TRUE ? (unsigned int)x.x_handy: 0);
 251 }
 252 
 253 unsigned int
 254 xdrrdma_authsize(AUTH *auth, struct cred *cred, int min_chunk)
 255 {
 256         XDR x;
 257         struct xdr_ops ops;
 258         bool_t stat;
 259         struct private *xdrp;
 260 
 261         x.x_ops = &ops;
 262         (void) xdrrdma_common(&x, min_chunk);
 263 
 264         stat = AUTH_MARSHALL(auth, &x, cred);
 265         xdrp = (struct private *)x.x_private;
 266         if (xdrp) {
 267                 if (xdrp->inline_buf)
 268                         mem_free(xdrp->inline_buf, xdrp->inline_len);
 269                 mem_free(xdrp, sizeof (struct private));
 270         }
 271         return (stat == TRUE ? (unsigned int)x.x_handy: 0);
 272 }
 273 
 274 struct xdr_ops *
 275 xdrrdma_xops(void)
 276 {
 277         static struct xdr_ops ops;
 278 
 279         ops.x_putbytes = x_putbytes;
 280         ops.x_inline = x_inline;
 281         ops.x_getpostn = x_getpostn;
 282         ops.x_setpostn = x_setpostn;
 283         ops.x_destroy = x_destroy;
 284         ops.x_control = x_control;
 285 
 286 #if defined(_LP64) || defined(_KERNEL)
 287         ops.x_getint32 = (void *)harmless;
 288         ops.x_putint32 = x_putint32_t;
 289 #endif
 290 
 291         /* the other harmless ones */
 292         ops.x_getbytes = (void *)harmless;
 293 
 294         return (&ops);
 295 }