xref: /illumos-gate/usr/src/uts/common/os/firmload.c (revision f73e0305a745f17c6a584c4470f99ea1e023657f)
1*f73e0305SHans Rosenfeld /*	$NetBSD: firmload.c,v 1.19 2014/03/25 16:19:13 christos Exp $	*/
2*f73e0305SHans Rosenfeld 
3*f73e0305SHans Rosenfeld /*
4*f73e0305SHans Rosenfeld  * Copyright 2016 Hans Rosenfeld <rosenfeld@grumpf.hope-2000.org>
5*f73e0305SHans Rosenfeld  */
6*f73e0305SHans Rosenfeld 
7*f73e0305SHans Rosenfeld /*
8*f73e0305SHans Rosenfeld  * Copyright (c) 2005, 2006 The NetBSD Foundation, Inc.
9*f73e0305SHans Rosenfeld  * All rights reserved.
10*f73e0305SHans Rosenfeld  *
11*f73e0305SHans Rosenfeld  * This code is derived from software contributed to The NetBSD Foundation
12*f73e0305SHans Rosenfeld  * by Jason R. Thorpe.
13*f73e0305SHans Rosenfeld  *
14*f73e0305SHans Rosenfeld  * Redistribution and use in source and binary forms, with or without
15*f73e0305SHans Rosenfeld  * modification, are permitted provided that the following conditions
16*f73e0305SHans Rosenfeld  * are met:
17*f73e0305SHans Rosenfeld  * 1. Redistributions of source code must retain the above copyright
18*f73e0305SHans Rosenfeld  *    notice, this list of conditions and the following disclaimer.
19*f73e0305SHans Rosenfeld  * 2. Redistributions in binary form must reproduce the above copyright
20*f73e0305SHans Rosenfeld  *    notice, this list of conditions and the following disclaimer in the
21*f73e0305SHans Rosenfeld  *    documentation and/or other materials provided with the distribution.
22*f73e0305SHans Rosenfeld  *
23*f73e0305SHans Rosenfeld  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
24*f73e0305SHans Rosenfeld  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
25*f73e0305SHans Rosenfeld  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
26*f73e0305SHans Rosenfeld  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
27*f73e0305SHans Rosenfeld  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
28*f73e0305SHans Rosenfeld  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
29*f73e0305SHans Rosenfeld  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
30*f73e0305SHans Rosenfeld  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
31*f73e0305SHans Rosenfeld  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
32*f73e0305SHans Rosenfeld  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
33*f73e0305SHans Rosenfeld  * POSSIBILITY OF SUCH DAMAGE.
34*f73e0305SHans Rosenfeld  */
35*f73e0305SHans Rosenfeld 
36*f73e0305SHans Rosenfeld /*
37*f73e0305SHans Rosenfeld  * The firmload API provides an interface for device drivers to access
38*f73e0305SHans Rosenfeld  * firmware images that must be loaded onto their devices.
39*f73e0305SHans Rosenfeld  */
40*f73e0305SHans Rosenfeld 
41*f73e0305SHans Rosenfeld #include <sys/param.h>
42*f73e0305SHans Rosenfeld #include <sys/fcntl.h>
43*f73e0305SHans Rosenfeld #include <sys/systm.h>
44*f73e0305SHans Rosenfeld #include <sys/vnode.h>
45*f73e0305SHans Rosenfeld #include <sys/lwp.h>
46*f73e0305SHans Rosenfeld #include <sys/file.h>
47*f73e0305SHans Rosenfeld #include <sys/cmn_err.h>
48*f73e0305SHans Rosenfeld #include <sys/modctl.h>
49*f73e0305SHans Rosenfeld #include <sys/kobj.h>
50*f73e0305SHans Rosenfeld #include <sys/kobj_impl.h>
51*f73e0305SHans Rosenfeld 
52*f73e0305SHans Rosenfeld #include <sys/firmload.h>
53*f73e0305SHans Rosenfeld 
54*f73e0305SHans Rosenfeld struct firmware_handle {
55*f73e0305SHans Rosenfeld 	struct _buf	*fh_buf;
56*f73e0305SHans Rosenfeld 	off_t		 fh_size;
57*f73e0305SHans Rosenfeld };
58*f73e0305SHans Rosenfeld 
59*f73e0305SHans Rosenfeld static firmware_handle_t
firmware_handle_alloc(void)60*f73e0305SHans Rosenfeld firmware_handle_alloc(void)
61*f73e0305SHans Rosenfeld {
62*f73e0305SHans Rosenfeld 	return (kmem_alloc(sizeof (struct firmware_handle), KM_SLEEP));
63*f73e0305SHans Rosenfeld }
64*f73e0305SHans Rosenfeld 
65*f73e0305SHans Rosenfeld static void
firmware_handle_free(firmware_handle_t fh)66*f73e0305SHans Rosenfeld firmware_handle_free(firmware_handle_t fh)
67*f73e0305SHans Rosenfeld {
68*f73e0305SHans Rosenfeld 	kmem_free(fh, sizeof (struct firmware_handle));
69*f73e0305SHans Rosenfeld }
70*f73e0305SHans Rosenfeld 
71*f73e0305SHans Rosenfeld /*
72*f73e0305SHans Rosenfeld  * firmware_open:
73*f73e0305SHans Rosenfeld  *
74*f73e0305SHans Rosenfeld  *	Open a firmware image and return its handle.
75*f73e0305SHans Rosenfeld  */
76*f73e0305SHans Rosenfeld int
firmware_open(const char * drvname,const char * imgname,firmware_handle_t * fhp)77*f73e0305SHans Rosenfeld firmware_open(const char *drvname, const char *imgname, firmware_handle_t *fhp)
78*f73e0305SHans Rosenfeld {
79*f73e0305SHans Rosenfeld 	char *path;
80*f73e0305SHans Rosenfeld 	firmware_handle_t fh;
81*f73e0305SHans Rosenfeld 	int error;
82*f73e0305SHans Rosenfeld 
83*f73e0305SHans Rosenfeld 	if (drvname == NULL || imgname == NULL || fhp == NULL)
84*f73e0305SHans Rosenfeld 		return (EINVAL);
85*f73e0305SHans Rosenfeld 
86*f73e0305SHans Rosenfeld 	path = kmem_asprintf("firmware/%s/%s", drvname, imgname);
87*f73e0305SHans Rosenfeld 	fh = firmware_handle_alloc();
88*f73e0305SHans Rosenfeld 
89*f73e0305SHans Rosenfeld 	fh->fh_buf = kobj_open_path(path, 1, 0);
90*f73e0305SHans Rosenfeld 	strfree(path);
91*f73e0305SHans Rosenfeld 
92*f73e0305SHans Rosenfeld 	if (fh->fh_buf == (struct _buf *)-1) {
93*f73e0305SHans Rosenfeld 		firmware_handle_free(fh);
94*f73e0305SHans Rosenfeld 		return (ENOENT);
95*f73e0305SHans Rosenfeld 	}
96*f73e0305SHans Rosenfeld 
97*f73e0305SHans Rosenfeld 	error = kobj_get_filesize(fh->fh_buf, (uint64_t *)&fh->fh_size);
98*f73e0305SHans Rosenfeld 	if (error != 0) {
99*f73e0305SHans Rosenfeld 		kobj_close_file(fh->fh_buf);
100*f73e0305SHans Rosenfeld 		firmware_handle_free(fh);
101*f73e0305SHans Rosenfeld 		return (error);
102*f73e0305SHans Rosenfeld 	}
103*f73e0305SHans Rosenfeld 
104*f73e0305SHans Rosenfeld 	*fhp = fh;
105*f73e0305SHans Rosenfeld 	return (0);
106*f73e0305SHans Rosenfeld }
107*f73e0305SHans Rosenfeld 
108*f73e0305SHans Rosenfeld /*
109*f73e0305SHans Rosenfeld  * firmware_close:
110*f73e0305SHans Rosenfeld  *
111*f73e0305SHans Rosenfeld  *	Close a firmware image.
112*f73e0305SHans Rosenfeld  */
113*f73e0305SHans Rosenfeld int
firmware_close(firmware_handle_t fh)114*f73e0305SHans Rosenfeld firmware_close(firmware_handle_t fh)
115*f73e0305SHans Rosenfeld {
116*f73e0305SHans Rosenfeld 	if (fh != NULL) {
117*f73e0305SHans Rosenfeld 		kobj_close_file(fh->fh_buf);
118*f73e0305SHans Rosenfeld 		firmware_handle_free(fh);
119*f73e0305SHans Rosenfeld 	}
120*f73e0305SHans Rosenfeld 	return (0);
121*f73e0305SHans Rosenfeld }
122*f73e0305SHans Rosenfeld 
123*f73e0305SHans Rosenfeld /*
124*f73e0305SHans Rosenfeld  * firmware_get_size:
125*f73e0305SHans Rosenfeld  *
126*f73e0305SHans Rosenfeld  *	Return the total size of a firmware image.
127*f73e0305SHans Rosenfeld  */
128*f73e0305SHans Rosenfeld off_t
firmware_get_size(firmware_handle_t fh)129*f73e0305SHans Rosenfeld firmware_get_size(firmware_handle_t fh)
130*f73e0305SHans Rosenfeld {
131*f73e0305SHans Rosenfeld 	ASSERT(fh != NULL);
132*f73e0305SHans Rosenfeld 	return (fh->fh_size);
133*f73e0305SHans Rosenfeld }
134*f73e0305SHans Rosenfeld 
135*f73e0305SHans Rosenfeld /*
136*f73e0305SHans Rosenfeld  * firmware_read:
137*f73e0305SHans Rosenfeld  *
138*f73e0305SHans Rosenfeld  *	Read data from a firmware image at the specified offset into
139*f73e0305SHans Rosenfeld  *	the provided buffer.
140*f73e0305SHans Rosenfeld  */
141*f73e0305SHans Rosenfeld int
firmware_read(firmware_handle_t fh,off_t offset,void * buf,size_t len)142*f73e0305SHans Rosenfeld firmware_read(firmware_handle_t fh, off_t offset, void *buf, size_t len)
143*f73e0305SHans Rosenfeld {
144*f73e0305SHans Rosenfeld 	ASSERT(fh != NULL);
145*f73e0305SHans Rosenfeld 	if (kobj_read_file(fh->fh_buf, buf, len, offset) == -1)
146*f73e0305SHans Rosenfeld 		return (-1);
147*f73e0305SHans Rosenfeld 
148*f73e0305SHans Rosenfeld 	return (0);
149*f73e0305SHans Rosenfeld }
150