2 * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
4 * Copyright (c) 2019 Justin Hibbits
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
9 * 1. Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
15 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
16 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
17 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
18 * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
19 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
20 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
21 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
23 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
24 * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
25 * POSSIBILITY OF SUCH DAMAGE.
30 #include <sys/param.h>
31 #include <sys/systm.h>
32 #include <sys/module.h>
36 #include <sys/kernel.h>
37 #include <sys/mutex.h>
40 #include <dev/ofw/openfirm.h>
41 #include <dev/ofw/ofw_bus.h>
42 #include <dev/ofw/ofw_bus_subr.h>
44 #include <machine/bus.h>
45 #include <machine/md_var.h>
46 #include <machine/pio.h>
47 #include <machine/resource.h>
56 #define NVRAM_BUFSIZE (65536) /* 64k blocks */
58 struct opal_nvram_softc {
63 vm_paddr_t sc_buf_phys;
69 #define NVRAM_LOCK(sc) mtx_lock(&sc->sc_mtx)
70 #define NVRAM_UNLOCK(sc) mtx_unlock(&sc->sc_mtx)
75 static int opal_nvram_probe(device_t);
76 static int opal_nvram_attach(device_t);
77 static int opal_nvram_detach(device_t);
82 static device_method_t opal_nvram_methods[] = {
83 /* Device interface */
84 DEVMETHOD(device_probe, opal_nvram_probe),
85 DEVMETHOD(device_attach, opal_nvram_attach),
86 DEVMETHOD(device_detach, opal_nvram_detach),
90 static driver_t opal_nvram_driver = {
93 sizeof(struct opal_nvram_softc)
96 static devclass_t opal_nvram_devclass;
98 DRIVER_MODULE(opal_nvram, opal, opal_nvram_driver, opal_nvram_devclass, 0, 0);
104 static d_open_t opal_nvram_open;
105 static d_close_t opal_nvram_close;
106 static d_read_t opal_nvram_read;
107 static d_write_t opal_nvram_write;
108 static d_ioctl_t opal_nvram_ioctl;
110 static struct cdevsw opal_nvram_cdevsw = {
111 .d_version = D_VERSION,
112 .d_open = opal_nvram_open,
113 .d_close = opal_nvram_close,
114 .d_read = opal_nvram_read,
115 .d_write = opal_nvram_write,
116 .d_ioctl = opal_nvram_ioctl,
121 opal_nvram_probe(device_t dev)
124 if (!ofw_bus_is_compatible(dev, "ibm,opal-nvram"))
127 device_set_desc(dev, "OPAL NVRAM");
128 return (BUS_PROBE_DEFAULT);
132 opal_nvram_attach(device_t dev)
134 struct opal_nvram_softc *sc;
138 node = ofw_bus_get_node(dev);
139 sc = device_get_softc(dev);
143 err = OF_getencprop(node, "#bytes", &sc->sc_size,
144 sizeof(sc->sc_size));
149 sc->sc_buf = contigmalloc(NVRAM_BUFSIZE, M_DEVBUF, M_WAITOK,
150 0, BUS_SPACE_MAXADDR, PAGE_SIZE, 0);
151 if (sc->sc_buf == NULL) {
152 device_printf(dev, "No memory for buffer.\n");
155 sc->sc_buf_phys = pmap_kextract((vm_offset_t)sc->sc_buf);
156 sc->sc_cdev = make_dev(&opal_nvram_cdevsw, 0, 0, 0, 0600,
158 sc->sc_cdev->si_drv1 = sc;
160 mtx_init(&sc->sc_mtx, "opal_nvram", 0, MTX_DEF);
166 opal_nvram_detach(device_t dev)
168 struct opal_nvram_softc *sc;
170 sc = device_get_softc(dev);
172 if (sc->sc_cdev != NULL)
173 destroy_dev(sc->sc_cdev);
174 if (sc->sc_buf != NULL)
175 contigfree(sc->sc_buf, NVRAM_BUFSIZE, M_DEVBUF);
177 mtx_destroy(&sc->sc_mtx);
183 opal_nvram_open(struct cdev *dev, int flags, int fmt, struct thread *td)
185 struct opal_nvram_softc *sc = dev->si_drv1;
201 opal_nvram_close(struct cdev *dev, int fflag, int devtype, struct thread *td)
203 struct opal_nvram_softc *sc = dev->si_drv1;
213 opal_nvram_read(struct cdev *dev, struct uio *uio, int ioflag)
215 struct opal_nvram_softc *sc = dev->si_drv1;
221 while (uio->uio_resid > 0) {
222 amnt = MIN(uio->uio_resid, sc->sc_size - uio->uio_offset);
223 amnt = MIN(amnt, NVRAM_BUFSIZE);
227 rv = opal_call(OPAL_READ_NVRAM, sc->sc_buf_phys,
228 amnt, uio->uio_offset);
229 if (rv != OPAL_SUCCESS) {
240 rv = uiomove(sc->sc_buf, amnt, uio);
250 opal_nvram_write(struct cdev *dev, struct uio *uio, int ioflag)
254 struct opal_nvram_softc *sc = dev->si_drv1;
259 while (uio->uio_resid > 0) {
260 amnt = MIN(uio->uio_resid, sc->sc_size - uio->uio_offset);
261 amnt = MIN(amnt, NVRAM_BUFSIZE);
266 offset = uio->uio_offset;
267 rv = uiomove(sc->sc_buf, amnt, uio);
270 rv = opal_call(OPAL_WRITE_NVRAM, sc->sc_buf_phys, amnt,
272 if (rv != OPAL_SUCCESS) {
291 opal_nvram_ioctl(struct cdev *dev, u_long cmd, caddr_t data, int fflag,
294 struct opal_nvram_softc *sc = dev->si_drv1;
298 *(off_t *)data = sc->sc_size;