xref: /freebsd/sys/dev/bhnd/nvram/bhnd_nvram_ioptr.c (revision 685dc743dc3b5645e34836464128e1c0558b404b)
1*f76db8deSLandon J. Fuller /*-
2*f76db8deSLandon J. Fuller  * Copyright (c) 2016 Landon Fuller <landonf@FreeBSD.org>
3*f76db8deSLandon J. Fuller  * All rights reserved.
4*f76db8deSLandon J. Fuller  *
5*f76db8deSLandon J. Fuller  * Redistribution and use in source and binary forms, with or without
6*f76db8deSLandon J. Fuller  * modification, are permitted provided that the following conditions
7*f76db8deSLandon J. Fuller  * are met:
8*f76db8deSLandon J. Fuller  * 1. Redistributions of source code must retain the above copyright
9*f76db8deSLandon J. Fuller  *    notice, this list of conditions and the following disclaimer,
10*f76db8deSLandon J. Fuller  *    without modification.
11*f76db8deSLandon J. Fuller  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
12*f76db8deSLandon J. Fuller  *    similar to the "NO WARRANTY" disclaimer below ("Disclaimer") and any
13*f76db8deSLandon J. Fuller  *    redistribution must be conditioned upon including a substantially
14*f76db8deSLandon J. Fuller  *    similar Disclaimer requirement for further binary redistribution.
15*f76db8deSLandon J. Fuller  *
16*f76db8deSLandon J. Fuller  * NO WARRANTY
17*f76db8deSLandon J. Fuller  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
18*f76db8deSLandon J. Fuller  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
19*f76db8deSLandon J. Fuller  * LIMITED TO, THE IMPLIED WARRANTIES OF NONINFRINGEMENT, MERCHANTIBILITY
20*f76db8deSLandon J. Fuller  * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
21*f76db8deSLandon J. Fuller  * THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY,
22*f76db8deSLandon J. Fuller  * OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23*f76db8deSLandon J. Fuller  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24*f76db8deSLandon J. Fuller  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
25*f76db8deSLandon J. Fuller  * IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26*f76db8deSLandon J. Fuller  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
27*f76db8deSLandon J. Fuller  * THE POSSIBILITY OF SUCH DAMAGES.
28*f76db8deSLandon J. Fuller  */
29*f76db8deSLandon J. Fuller 
30*f76db8deSLandon J. Fuller #include <sys/cdefs.h>
31*f76db8deSLandon J. Fuller #ifdef _KERNEL
32*f76db8deSLandon J. Fuller #include <sys/param.h>
33*f76db8deSLandon J. Fuller #include <sys/malloc.h>
34*f76db8deSLandon J. Fuller #include <sys/systm.h>
35*f76db8deSLandon J. Fuller #else /* !_KERNEL */
36*f76db8deSLandon J. Fuller #include <errno.h>
37*f76db8deSLandon J. Fuller #include <stdint.h>
38*f76db8deSLandon J. Fuller #include <stdlib.h>
39*f76db8deSLandon J. Fuller #include <string.h>
40*f76db8deSLandon J. Fuller #endif /* _KERNEL */
41*f76db8deSLandon J. Fuller 
42*f76db8deSLandon J. Fuller #include "bhnd_nvram_private.h"
43*f76db8deSLandon J. Fuller 
44*f76db8deSLandon J. Fuller #include "bhnd_nvram_io.h"
45*f76db8deSLandon J. Fuller #include "bhnd_nvram_iovar.h"
46*f76db8deSLandon J. Fuller 
47*f76db8deSLandon J. Fuller /**
48*f76db8deSLandon J. Fuller  * Memory-backed NVRAM I/O context.
49*f76db8deSLandon J. Fuller  *
50*f76db8deSLandon J. Fuller  * ioptr instances are gauranteed to provide persistent references to its
51*f76db8deSLandon J. Fuller  * backing contigious memory via bhnd_nvram_io_read_ptr() and
52*f76db8deSLandon J. Fuller  * bhnd_nvram_io_write_ptr().
53*f76db8deSLandon J. Fuller  */
54*f76db8deSLandon J. Fuller struct bhnd_nvram_ioptr {
55*f76db8deSLandon J. Fuller 	struct bhnd_nvram_io	 io;		/**< common I/O instance state */
56*f76db8deSLandon J. Fuller 	void			*ptr;		/**< backing memory */
57*f76db8deSLandon J. Fuller 	size_t			 size;		/**< size at @p ptr */
58*f76db8deSLandon J. Fuller 	size_t			 capacity;	/**< capacity at @p ptr */
59*f76db8deSLandon J. Fuller 	uint32_t		 flags;		/**< flags (see BHND_NVRAM_IOPTR_*) */
60*f76db8deSLandon J. Fuller };
61*f76db8deSLandon J. Fuller 
BHND_NVRAM_IOPS_DEFN(ioptr)62*f76db8deSLandon J. Fuller BHND_NVRAM_IOPS_DEFN(ioptr)
63*f76db8deSLandon J. Fuller 
64*f76db8deSLandon J. Fuller /**
65*f76db8deSLandon J. Fuller  * Allocate and return a new I/O context, mapping @p size bytes at @p ptr.
66*f76db8deSLandon J. Fuller 
67*f76db8deSLandon J. Fuller  * The caller is responsible for deallocating the returned I/O context via
68*f76db8deSLandon J. Fuller  * bhnd_nvram_io_free().
69*f76db8deSLandon J. Fuller  *
70*f76db8deSLandon J. Fuller  * @param	ptr		The pointer to be mapped by the returned I/O
71*f76db8deSLandon J. Fuller  *				context. Must remain valid for the lifetime of
72*f76db8deSLandon J. Fuller  *				the returned I/O context.
73*f76db8deSLandon J. Fuller  * @param	size		The total number of bytes mapped at @p ptr.
74*f76db8deSLandon J. Fuller  * @param	capacity	The maximum number of bytes that may be mapped
75*f76db8deSLandon J. Fuller  *				at @p ptr via bhnd_nvram_ioptr_setsize().
76*f76db8deSLandon J. Fuller  * @param	flags		Access flags (see BHND_NVRAM_IOPTR_*).
77*f76db8deSLandon J. Fuller  *
78*f76db8deSLandon J. Fuller  * @retval	bhnd_nvram_io	success.
79*f76db8deSLandon J. Fuller  * @retval	NULL		allocation failed.
80*f76db8deSLandon J. Fuller  * @retval	NULL		the requested @p capacity is less than @p size.
81*f76db8deSLandon J. Fuller  */
82*f76db8deSLandon J. Fuller struct bhnd_nvram_io *
83*f76db8deSLandon J. Fuller bhnd_nvram_ioptr_new(const void *ptr, size_t size, size_t capacity,
84*f76db8deSLandon J. Fuller     uint32_t flags)
85*f76db8deSLandon J. Fuller {
86*f76db8deSLandon J. Fuller 	struct bhnd_nvram_ioptr	*ioptr;
87*f76db8deSLandon J. Fuller 
88*f76db8deSLandon J. Fuller 	/* Sanity check the capacity */
89*f76db8deSLandon J. Fuller 	if (size > capacity)
90*f76db8deSLandon J. Fuller 		return (NULL);
91*f76db8deSLandon J. Fuller 
92*f76db8deSLandon J. Fuller 	/* Allocate I/O context */
93*f76db8deSLandon J. Fuller 	ioptr = bhnd_nv_malloc(sizeof(*ioptr));
94*f76db8deSLandon J. Fuller 	if (ioptr == NULL)
95*f76db8deSLandon J. Fuller 		return (NULL);
96*f76db8deSLandon J. Fuller 
97*f76db8deSLandon J. Fuller 	ioptr->io.iops = &bhnd_nvram_ioptr_ops;
98*f76db8deSLandon J. Fuller 	ioptr->ptr = __DECONST(void *, ptr);
99*f76db8deSLandon J. Fuller 	ioptr->size = size;
100*f76db8deSLandon J. Fuller 	ioptr->capacity = capacity;
101*f76db8deSLandon J. Fuller 	ioptr->flags = flags;
102*f76db8deSLandon J. Fuller 
103*f76db8deSLandon J. Fuller 	return (&ioptr->io);
104*f76db8deSLandon J. Fuller }
105*f76db8deSLandon J. Fuller 
106*f76db8deSLandon J. Fuller static void
bhnd_nvram_ioptr_free(struct bhnd_nvram_io * io)107*f76db8deSLandon J. Fuller bhnd_nvram_ioptr_free(struct bhnd_nvram_io *io)
108*f76db8deSLandon J. Fuller {
109*f76db8deSLandon J. Fuller 	bhnd_nv_free(io);
110*f76db8deSLandon J. Fuller }
111*f76db8deSLandon J. Fuller 
112*f76db8deSLandon J. Fuller static size_t
bhnd_nvram_ioptr_getsize(struct bhnd_nvram_io * io)113*f76db8deSLandon J. Fuller bhnd_nvram_ioptr_getsize(struct bhnd_nvram_io *io)
114*f76db8deSLandon J. Fuller {
115*f76db8deSLandon J. Fuller 	struct bhnd_nvram_ioptr	*ioptr = (struct bhnd_nvram_ioptr *)io;
116*f76db8deSLandon J. Fuller 	return (ioptr->size);
117*f76db8deSLandon J. Fuller }
118*f76db8deSLandon J. Fuller 
119*f76db8deSLandon J. Fuller static int
bhnd_nvram_ioptr_setsize(struct bhnd_nvram_io * io,size_t size)120*f76db8deSLandon J. Fuller bhnd_nvram_ioptr_setsize(struct bhnd_nvram_io *io, size_t size)
121*f76db8deSLandon J. Fuller {
122*f76db8deSLandon J. Fuller 	struct bhnd_nvram_ioptr	*ioptr = (struct bhnd_nvram_ioptr *)io;
123*f76db8deSLandon J. Fuller 
124*f76db8deSLandon J. Fuller 	/* Must be writable */
125*f76db8deSLandon J. Fuller 	if (!(ioptr->flags & BHND_NVRAM_IOPTR_RDWR))
126*f76db8deSLandon J. Fuller 		return (ENODEV);
127*f76db8deSLandon J. Fuller 
128*f76db8deSLandon J. Fuller 	/* Can't exceed the actual capacity */
129*f76db8deSLandon J. Fuller 	if (size > ioptr->capacity)
130*f76db8deSLandon J. Fuller 		return (ENXIO);
131*f76db8deSLandon J. Fuller 
132*f76db8deSLandon J. Fuller 	ioptr->size = size;
133*f76db8deSLandon J. Fuller 	return (0);
134*f76db8deSLandon J. Fuller }
135*f76db8deSLandon J. Fuller 
136*f76db8deSLandon J. Fuller /* Common ioptr_(read|write)_ptr implementation */
137*f76db8deSLandon J. Fuller static int
bhnd_nvram_ioptr_ptr(struct bhnd_nvram_ioptr * ioptr,size_t offset,void ** ptr,size_t nbytes,size_t * navail)138*f76db8deSLandon J. Fuller bhnd_nvram_ioptr_ptr(struct bhnd_nvram_ioptr *ioptr, size_t offset, void **ptr,
139*f76db8deSLandon J. Fuller 		     size_t nbytes, size_t *navail)
140*f76db8deSLandon J. Fuller {
141*f76db8deSLandon J. Fuller 	size_t avail;
142*f76db8deSLandon J. Fuller 
143*f76db8deSLandon J. Fuller 	/* Verify offset+nbytes fall within the buffer range */
144*f76db8deSLandon J. Fuller 	if (offset > ioptr->size)
145*f76db8deSLandon J. Fuller 		return (ENXIO);
146*f76db8deSLandon J. Fuller 
147*f76db8deSLandon J. Fuller 	avail = ioptr->size - offset;
148*f76db8deSLandon J. Fuller 	if (avail < nbytes)
149*f76db8deSLandon J. Fuller 		return (ENXIO);
150*f76db8deSLandon J. Fuller 
151*f76db8deSLandon J. Fuller 	/* Valid I/O range, provide a pointer to the buffer and the
152*f76db8deSLandon J. Fuller 	 * total count of available bytes */
153*f76db8deSLandon J. Fuller 	*ptr = ((uint8_t *)ioptr->ptr) + offset;
154*f76db8deSLandon J. Fuller 	if (navail != NULL)
155*f76db8deSLandon J. Fuller 		*navail = avail;
156*f76db8deSLandon J. Fuller 
157*f76db8deSLandon J. Fuller 	return (0);
158*f76db8deSLandon J. Fuller }
159*f76db8deSLandon J. Fuller 
160*f76db8deSLandon J. Fuller static int
bhnd_nvram_ioptr_read_ptr(struct bhnd_nvram_io * io,size_t offset,const void ** ptr,size_t nbytes,size_t * navail)161*f76db8deSLandon J. Fuller bhnd_nvram_ioptr_read_ptr(struct bhnd_nvram_io *io, size_t offset,
162*f76db8deSLandon J. Fuller 			  const void **ptr, size_t nbytes, size_t *navail)
163*f76db8deSLandon J. Fuller {
164*f76db8deSLandon J. Fuller 	struct bhnd_nvram_ioptr	*ioptr;
165*f76db8deSLandon J. Fuller 	void			*writep;
166*f76db8deSLandon J. Fuller 	int			 error;
167*f76db8deSLandon J. Fuller 
168*f76db8deSLandon J. Fuller 	ioptr = (struct bhnd_nvram_ioptr *) io;
169*f76db8deSLandon J. Fuller 
170*f76db8deSLandon J. Fuller 	/* Return a pointer into our backing buffer */
171*f76db8deSLandon J. Fuller 	error = bhnd_nvram_ioptr_ptr(ioptr, offset, &writep, nbytes, navail);
172*f76db8deSLandon J. Fuller 	if (error)
173*f76db8deSLandon J. Fuller 		return (error);
174*f76db8deSLandon J. Fuller 
175*f76db8deSLandon J. Fuller 	*ptr = writep;
176*f76db8deSLandon J. Fuller 
177*f76db8deSLandon J. Fuller 	return (0);
178*f76db8deSLandon J. Fuller }
179*f76db8deSLandon J. Fuller 
180*f76db8deSLandon J. Fuller static int
bhnd_nvram_ioptr_write_ptr(struct bhnd_nvram_io * io,size_t offset,void ** ptr,size_t nbytes,size_t * navail)181*f76db8deSLandon J. Fuller bhnd_nvram_ioptr_write_ptr(struct bhnd_nvram_io *io, size_t offset,
182*f76db8deSLandon J. Fuller 			   void **ptr, size_t nbytes, size_t *navail)
183*f76db8deSLandon J. Fuller {
184*f76db8deSLandon J. Fuller 	struct bhnd_nvram_ioptr	*ioptr;
185*f76db8deSLandon J. Fuller 
186*f76db8deSLandon J. Fuller 	ioptr = (struct bhnd_nvram_ioptr *) io;
187*f76db8deSLandon J. Fuller 
188*f76db8deSLandon J. Fuller 	/* Must be writable */
189*f76db8deSLandon J. Fuller 	if (!(ioptr->flags & BHND_NVRAM_IOPTR_RDWR))
190*f76db8deSLandon J. Fuller 		return (ENODEV);
191*f76db8deSLandon J. Fuller 
192*f76db8deSLandon J. Fuller 	/* Return a pointer into our backing buffer */
193*f76db8deSLandon J. Fuller 	return (bhnd_nvram_ioptr_ptr(ioptr, offset, ptr, nbytes, navail));
194*f76db8deSLandon J. Fuller }
195*f76db8deSLandon J. Fuller 
196*f76db8deSLandon J. Fuller static int
bhnd_nvram_ioptr_read(struct bhnd_nvram_io * io,size_t offset,void * buffer,size_t nbytes)197*f76db8deSLandon J. Fuller bhnd_nvram_ioptr_read(struct bhnd_nvram_io *io, size_t offset, void *buffer,
198*f76db8deSLandon J. Fuller 		      size_t nbytes)
199*f76db8deSLandon J. Fuller {
200*f76db8deSLandon J. Fuller 	const void	*ptr;
201*f76db8deSLandon J. Fuller 	int		 error;
202*f76db8deSLandon J. Fuller 
203*f76db8deSLandon J. Fuller 	/* Try to fetch a direct pointer for at least nbytes */
204*f76db8deSLandon J. Fuller 	if ((error = bhnd_nvram_io_read_ptr(io, offset, &ptr, nbytes, NULL)))
205*f76db8deSLandon J. Fuller 		return (error);
206*f76db8deSLandon J. Fuller 
207*f76db8deSLandon J. Fuller 	/* Copy out the requested data */
208*f76db8deSLandon J. Fuller 	memcpy(buffer, ptr, nbytes);
209*f76db8deSLandon J. Fuller 	return (0);
210*f76db8deSLandon J. Fuller }
211*f76db8deSLandon J. Fuller 
212*f76db8deSLandon J. Fuller static int
bhnd_nvram_ioptr_write(struct bhnd_nvram_io * io,size_t offset,void * buffer,size_t nbytes)213*f76db8deSLandon J. Fuller bhnd_nvram_ioptr_write(struct bhnd_nvram_io *io, size_t offset,
214*f76db8deSLandon J. Fuller 		       void *buffer, size_t nbytes)
215*f76db8deSLandon J. Fuller {
216*f76db8deSLandon J. Fuller 	void	*ptr;
217*f76db8deSLandon J. Fuller 	int	 error;
218*f76db8deSLandon J. Fuller 
219*f76db8deSLandon J. Fuller 	/* Try to fetch a direct pointer for at least nbytes */
220*f76db8deSLandon J. Fuller 	if ((error = bhnd_nvram_io_write_ptr(io, offset, &ptr, nbytes, NULL)))
221*f76db8deSLandon J. Fuller 		return (error);
222*f76db8deSLandon J. Fuller 
223*f76db8deSLandon J. Fuller 	/* Copy in the provided data */
224*f76db8deSLandon J. Fuller 	memcpy(ptr, buffer, nbytes);
225*f76db8deSLandon J. Fuller 	return (0);
226*f76db8deSLandon J. Fuller }
227