1 /*
2 * xdr_sizeof.c
3 *
4 * Copyright (c) 2010, Oracle America, Inc.
5 *
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions are
8 * met:
9 *
10 * * Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * * Redistributions in binary form must reproduce the above
13 * copyright notice, this list of conditions and the following
14 * disclaimer in the documentation and/or other materials
15 * provided with the distribution.
16 * * Neither the name of the "Oracle America, Inc." nor the names of its
17 * contributors may be used to endorse or promote products derived
18 * from this software without specific prior written permission.
19 *
20 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
21 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
23 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
24 * COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
25 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
27 * GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
28 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
29 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
30 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
31 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32 *
33 * General purpose routine to see how much space something will use
34 * when serialized using XDR.
35 */
36
37 #include <rpc/types.h>
38 #include <rpc/xdr.h>
39 #include <sys/types.h>
40 #include <stdlib.h>
41 #include <shlib-compat.h>
42
43 /* ARGSUSED */
44 static bool_t
x_putlong(XDR * xdrs,const long * longp)45 x_putlong (XDR *xdrs, const long *longp)
46 {
47 xdrs->x_handy += BYTES_PER_XDR_UNIT;
48 return TRUE;
49 }
50
51 /* ARGSUSED */
52 static bool_t
x_putbytes(XDR * xdrs,const char * bp,u_int len)53 x_putbytes (XDR *xdrs, const char *bp, u_int len)
54 {
55 xdrs->x_handy += len;
56 return TRUE;
57 }
58
59 static u_int
x_getpostn(const XDR * xdrs)60 x_getpostn (const XDR *xdrs)
61 {
62 return xdrs->x_handy;
63 }
64
65 /* ARGSUSED */
66 static bool_t
x_setpostn(XDR * xdrs,u_int len)67 x_setpostn (XDR *xdrs, u_int len)
68 {
69 /* This is not allowed */
70 return FALSE;
71 }
72
73 static int32_t *
x_inline(XDR * xdrs,u_int len)74 x_inline (XDR *xdrs, u_int len)
75 {
76 if (len == 0)
77 return NULL;
78 if (xdrs->x_op != XDR_ENCODE)
79 return NULL;
80 if (len < (u_int) (long int) xdrs->x_base)
81 {
82 /* x_private was already allocated */
83 xdrs->x_handy += len;
84 return (int32_t *) xdrs->x_private;
85 }
86 else
87 {
88 /* Free the earlier space and allocate new area */
89 free (xdrs->x_private);
90 if ((xdrs->x_private = (caddr_t) malloc (len)) == NULL)
91 {
92 xdrs->x_base = 0;
93 return NULL;
94 }
95 xdrs->x_base = (void *) (long) len;
96 xdrs->x_handy += len;
97 return (int32_t *) xdrs->x_private;
98 }
99 }
100
101 static int
harmless(void)102 harmless (void)
103 {
104 /* Always return FALSE/NULL, as the case may be */
105 return 0;
106 }
107
108 static void
x_destroy(XDR * xdrs)109 x_destroy (XDR *xdrs)
110 {
111 xdrs->x_handy = 0;
112 xdrs->x_base = 0;
113 if (xdrs->x_private)
114 {
115 free (xdrs->x_private);
116 xdrs->x_private = NULL;
117 }
118 return;
119 }
120
121 static bool_t
x_putint32(XDR * xdrs,const int32_t * int32p)122 x_putint32 (XDR *xdrs, const int32_t *int32p)
123 {
124 xdrs->x_handy += BYTES_PER_XDR_UNIT;
125 return TRUE;
126 }
127
128 unsigned long
xdr_sizeof(xdrproc_t func,void * data)129 xdr_sizeof (xdrproc_t func, void *data)
130 {
131 XDR x;
132 struct xdr_ops ops;
133 bool_t stat;
134 /* to stop ANSI-C compiler from complaining */
135 typedef bool_t (*dummyfunc1) (XDR *, long *);
136 typedef bool_t (*dummyfunc2) (XDR *, caddr_t, u_int);
137 typedef bool_t (*dummyfunc3) (XDR *, int32_t *);
138
139 ops.x_putlong = x_putlong;
140 ops.x_putbytes = x_putbytes;
141 ops.x_inline = x_inline;
142 ops.x_getpostn = x_getpostn;
143 ops.x_setpostn = x_setpostn;
144 ops.x_destroy = x_destroy;
145 ops.x_putint32 = x_putint32;
146
147 /* the other harmless ones */
148 ops.x_getlong = (dummyfunc1) harmless;
149 ops.x_getbytes = (dummyfunc2) harmless;
150 ops.x_getint32 = (dummyfunc3) harmless;
151
152 x.x_op = XDR_ENCODE;
153 x.x_ops = &ops;
154 x.x_handy = 0;
155 x.x_private = (caddr_t) NULL;
156 x.x_base = (caddr_t) 0;
157
158 stat = func (&x, data);
159 free (x.x_private);
160 return stat == TRUE ? x.x_handy : 0;
161 }
162 #ifdef EXPORT_RPC_SYMBOLS
163 libc_hidden_def (xdr_sizeof)
164 #else
165 libc_hidden_nolink_sunrpc (xdr_sizeof, GLIBC_2_1)
166 #endif
167