xref: /xnu-11215.1.10/iokit/bsddev/DINetBootHook.cpp (revision 8d741a5de7ff4191bf97d57b9f54c2f6d4a15585)
1*8d741a5dSApple OSS Distributions /*
2*8d741a5dSApple OSS Distributions  * Copyright (c) 2002-2016 Apple Inc. All rights reserved.
3*8d741a5dSApple OSS Distributions  *
4*8d741a5dSApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5*8d741a5dSApple OSS Distributions  *
6*8d741a5dSApple OSS Distributions  * This file contains Original Code and/or Modifications of Original Code
7*8d741a5dSApple OSS Distributions  * as defined in and that are subject to the Apple Public Source License
8*8d741a5dSApple OSS Distributions  * Version 2.0 (the 'License'). You may not use this file except in
9*8d741a5dSApple OSS Distributions  * compliance with the License. The rights granted to you under the License
10*8d741a5dSApple OSS Distributions  * may not be used to create, or enable the creation or redistribution of,
11*8d741a5dSApple OSS Distributions  * unlawful or unlicensed copies of an Apple operating system, or to
12*8d741a5dSApple OSS Distributions  * circumvent, violate, or enable the circumvention or violation of, any
13*8d741a5dSApple OSS Distributions  * terms of an Apple operating system software license agreement.
14*8d741a5dSApple OSS Distributions  *
15*8d741a5dSApple OSS Distributions  * Please obtain a copy of the License at
16*8d741a5dSApple OSS Distributions  * http://www.opensource.apple.com/apsl/ and read it before using this file.
17*8d741a5dSApple OSS Distributions  *
18*8d741a5dSApple OSS Distributions  * The Original Code and all software distributed under the License are
19*8d741a5dSApple OSS Distributions  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20*8d741a5dSApple OSS Distributions  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21*8d741a5dSApple OSS Distributions  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22*8d741a5dSApple OSS Distributions  * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23*8d741a5dSApple OSS Distributions  * Please see the License for the specific language governing rights and
24*8d741a5dSApple OSS Distributions  * limitations under the License.
25*8d741a5dSApple OSS Distributions  *
26*8d741a5dSApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27*8d741a5dSApple OSS Distributions  */
28*8d741a5dSApple OSS Distributions /*
29*8d741a5dSApple OSS Distributions  *  DINetBootHook.c
30*8d741a5dSApple OSS Distributions  *  DiskImages
31*8d741a5dSApple OSS Distributions  *
32*8d741a5dSApple OSS Distributions  *  Created by Byron Han on Sat Apr 13 2002.
33*8d741a5dSApple OSS Distributions  *
34*8d741a5dSApple OSS Distributions  *	Revision History
35*8d741a5dSApple OSS Distributions  *
36*8d741a5dSApple OSS Distributions  *	$Log: DINetBootHook.cpp,v $
37*8d741a5dSApple OSS Distributions  *	Revision 1.4  2005/07/29 21:49:57  lindak
38*8d741a5dSApple OSS Distributions  *	Merge of branch "chardonnay" to pick up all chardonnay changes in Leopard
39*8d741a5dSApple OSS Distributions  *	as of xnu-792.7.4
40*8d741a5dSApple OSS Distributions  *
41*8d741a5dSApple OSS Distributions  *	Revision 1.3.1558.1  2005/06/24 01:47:25  lindak
42*8d741a5dSApple OSS Distributions  *	Bringing over all of the Karma changes into chardonnay.
43*8d741a5dSApple OSS Distributions  *
44*8d741a5dSApple OSS Distributions  *	Revision 1.1.1.1  2005/02/24 21:48:06  akosut
45*8d741a5dSApple OSS Distributions  *	Import xnu-764 from Tiger8A395
46*8d741a5dSApple OSS Distributions  *
47*8d741a5dSApple OSS Distributions  *	Revision 1.3  2002/06/16 20:36:02  lindak
48*8d741a5dSApple OSS Distributions  *	Merged PR-2957314 into Jaguar (siegmund: netboot kernel code needs to set
49*8d741a5dSApple OSS Distributions  *	com.apple.AppleDiskImageController.load to boolean Yes)
50*8d741a5dSApple OSS Distributions  *
51*8d741a5dSApple OSS Distributions  *	Revision 1.2.40.2  2002/06/15 03:50:38  dieter
52*8d741a5dSApple OSS Distributions  *	- corrected com.apple.AppleDiskImageController.load string
53*8d741a5dSApple OSS Distributions  *
54*8d741a5dSApple OSS Distributions  *	Revision 1.2.40.1  2002/06/15 03:01:08  dieter
55*8d741a5dSApple OSS Distributions  *	Bug #: 2957314
56*8d741a5dSApple OSS Distributions  *	- add call to force IOHDIXController to get loaded/matched
57*8d741a5dSApple OSS Distributions  *
58*8d741a5dSApple OSS Distributions  *	Revision 1.2  2002/05/03 18:08:39  lindak
59*8d741a5dSApple OSS Distributions  *	Merged PR-2909558 into Jaguar (siegmund POST WWDC: add support for NetBoot
60*8d741a5dSApple OSS Distributions  *	over IOHDIXController)
61*8d741a5dSApple OSS Distributions  *
62*8d741a5dSApple OSS Distributions  *	Revision 1.1.2.1  2002/04/24 22:29:12  dieter
63*8d741a5dSApple OSS Distributions  *	Bug #: 2909558
64*8d741a5dSApple OSS Distributions  *	- added IOHDIXController netboot stubs
65*8d741a5dSApple OSS Distributions  *
66*8d741a5dSApple OSS Distributions  *	Revision 1.3  2002/04/16 00:41:37  han
67*8d741a5dSApple OSS Distributions  *	migrated code out of here to IOHDIXController's setProperty method
68*8d741a5dSApple OSS Distributions  *
69*8d741a5dSApple OSS Distributions  *	Revision 1.2  2002/04/14 23:53:53  han
70*8d741a5dSApple OSS Distributions  *	eliminate qDEBUG=1, use emums instead of hard coded string constants
71*8d741a5dSApple OSS Distributions  *
72*8d741a5dSApple OSS Distributions  *	Revision 1.1  2002/04/14 22:54:42  han
73*8d741a5dSApple OSS Distributions  *	Renamed from DINetBookHook.c.
74*8d741a5dSApple OSS Distributions  *	First stab at implementing this code.
75*8d741a5dSApple OSS Distributions  *
76*8d741a5dSApple OSS Distributions  *	Revision 1.1  2002/04/13 19:22:28  han
77*8d741a5dSApple OSS Distributions  *	added stub file DINetBookHook.c
78*8d741a5dSApple OSS Distributions  *
79*8d741a5dSApple OSS Distributions  *
80*8d741a5dSApple OSS Distributions  */
81*8d741a5dSApple OSS Distributions #ifndef qDEBUG
82*8d741a5dSApple OSS Distributions #define qDEBUG 0
83*8d741a5dSApple OSS Distributions #endif
84*8d741a5dSApple OSS Distributions 
85*8d741a5dSApple OSS Distributions #if qDEBUG
86*8d741a5dSApple OSS Distributions #warning qDEBUG is 1!
87*8d741a5dSApple OSS Distributions #endif
88*8d741a5dSApple OSS Distributions 
89*8d741a5dSApple OSS Distributions #include <sys/types.h>
90*8d741a5dSApple OSS Distributions #include <mach/clock_types.h>
91*8d741a5dSApple OSS Distributions #include <IOKit/IOService.h>
92*8d741a5dSApple OSS Distributions #include <IOKit/IOLib.h>
93*8d741a5dSApple OSS Distributions #include "DINetBootHook.h"
94*8d741a5dSApple OSS Distributions 
95*8d741a5dSApple OSS Distributions #define kIOHDIXControllerClassName      "IOHDIXController"
96*8d741a5dSApple OSS Distributions #define kDIRootImageKey                         "di-root-image"
97*8d741a5dSApple OSS Distributions #define kDIRootImageRemovableKey                "di-root-removable"
98*8d741a5dSApple OSS Distributions #define kDIRootImageResultKey           "di-root-image-result"
99*8d741a5dSApple OSS Distributions #define kDIRootImageDevNameKey          "di-root-image-devname"
100*8d741a5dSApple OSS Distributions #define kDIRootImageDevTKey                     "di-root-image-devt"
101*8d741a5dSApple OSS Distributions #define kDIRootRamFileKey           "di-root-ram-file"
102*8d741a5dSApple OSS Distributions 
103*8d741a5dSApple OSS Distributions #define kDIMatchQuiesceTimeout          30ull
104*8d741a5dSApple OSS Distributions 
105*8d741a5dSApple OSS Distributions static IOService *
di_load_controller(void)106*8d741a5dSApple OSS Distributions di_load_controller( void )
107*8d741a5dSApple OSS Distributions {
108*8d741a5dSApple OSS Distributions 	OSIterator *    controllerIterator      = NULL;
109*8d741a5dSApple OSS Distributions 	OSDictionary *  matchDictionary         = NULL;
110*8d741a5dSApple OSS Distributions 	IOService *     controller                      = NULL;
111*8d741a5dSApple OSS Distributions 
112*8d741a5dSApple OSS Distributions 	do {
113*8d741a5dSApple OSS Distributions 		IOService::getResourceService()->publishResource("com.apple.AppleDiskImageController.load", kOSBooleanTrue);
114*8d741a5dSApple OSS Distributions 		IOService::getResourceService()->waitQuiet();
115*8d741a5dSApple OSS Distributions 
116*8d741a5dSApple OSS Distributions 		// first find IOHDIXController
117*8d741a5dSApple OSS Distributions 		matchDictionary = IOService::serviceMatching(kIOHDIXControllerClassName);
118*8d741a5dSApple OSS Distributions 		if (!matchDictionary) {
119*8d741a5dSApple OSS Distributions 			break;
120*8d741a5dSApple OSS Distributions 		}
121*8d741a5dSApple OSS Distributions 
122*8d741a5dSApple OSS Distributions 		controllerIterator = IOService::getMatchingServices(matchDictionary);
123*8d741a5dSApple OSS Distributions 		if (!controllerIterator) {
124*8d741a5dSApple OSS Distributions 			break;
125*8d741a5dSApple OSS Distributions 		}
126*8d741a5dSApple OSS Distributions 
127*8d741a5dSApple OSS Distributions 		controller = OSDynamicCast(IOService, controllerIterator->getNextObject());
128*8d741a5dSApple OSS Distributions 		if (!controller) {
129*8d741a5dSApple OSS Distributions 			break;
130*8d741a5dSApple OSS Distributions 		}
131*8d741a5dSApple OSS Distributions 
132*8d741a5dSApple OSS Distributions 		controller->retain();
133*8d741a5dSApple OSS Distributions 	} while (false);
134*8d741a5dSApple OSS Distributions 
135*8d741a5dSApple OSS Distributions 	if (matchDictionary) {
136*8d741a5dSApple OSS Distributions 		matchDictionary->release();
137*8d741a5dSApple OSS Distributions 	}
138*8d741a5dSApple OSS Distributions 	if (controllerIterator) {
139*8d741a5dSApple OSS Distributions 		controllerIterator->release();
140*8d741a5dSApple OSS Distributions 	}
141*8d741a5dSApple OSS Distributions 
142*8d741a5dSApple OSS Distributions 	return controller;
143*8d741a5dSApple OSS Distributions }
144*8d741a5dSApple OSS Distributions 
145*8d741a5dSApple OSS Distributions extern "C" {
146*8d741a5dSApple OSS Distributions /* FIXME: removable should be replaced with a struct (so it could be easily
147*8d741a5dSApple OSS Distributions  * extensible in the future). However, since there is no common header file
148*8d741a5dSApple OSS Distributions  * between imageboot and NetBoot, we opt for a simple bool for now.
149*8d741a5dSApple OSS Distributions  * Refactor this into a common header file.
150*8d741a5dSApple OSS Distributions  */
151*8d741a5dSApple OSS Distributions static int
di_add_properties(IOService * controller,bool removable)152*8d741a5dSApple OSS Distributions di_add_properties(IOService *controller, bool removable)
153*8d741a5dSApple OSS Distributions {
154*8d741a5dSApple OSS Distributions 	if (!controller->setProperty(kDIRootImageRemovableKey, removable ? kOSBooleanTrue : kOSBooleanFalse)) {
155*8d741a5dSApple OSS Distributions 		IOLog("IOHDIXController::setProperty(%s, %d) failed.\n", kDIRootImageRemovableKey, !!removable);
156*8d741a5dSApple OSS Distributions 		return kIOReturnBadArgument;
157*8d741a5dSApple OSS Distributions 	}
158*8d741a5dSApple OSS Distributions 
159*8d741a5dSApple OSS Distributions 	return kIOReturnSuccess;
160*8d741a5dSApple OSS Distributions }
161*8d741a5dSApple OSS Distributions 
162*8d741a5dSApple OSS Distributions int
di_root_image_ext(const char * path,char * devname,size_t devsz,dev_t * dev_p,bool removable)163*8d741a5dSApple OSS Distributions di_root_image_ext(const char *path, char *devname, size_t devsz, dev_t *dev_p, bool removable)
164*8d741a5dSApple OSS Distributions {
165*8d741a5dSApple OSS Distributions 	IOReturn                        res                             = 0;
166*8d741a5dSApple OSS Distributions 	IOService               *       controller                      = NULL;
167*8d741a5dSApple OSS Distributions 	OSString                *       pathString                      = NULL;
168*8d741a5dSApple OSS Distributions 	OSNumber                *       myResult                        = NULL;
169*8d741a5dSApple OSS Distributions 	OSString                *       myDevName                       = NULL;
170*8d741a5dSApple OSS Distributions 	OSNumber                *       myDevT                          = NULL;
171*8d741a5dSApple OSS Distributions 
172*8d741a5dSApple OSS Distributions 	// sanity check arguments please
173*8d741a5dSApple OSS Distributions 	if (devname) {
174*8d741a5dSApple OSS Distributions 		*devname = 0;
175*8d741a5dSApple OSS Distributions 	}
176*8d741a5dSApple OSS Distributions 	if (dev_p) {
177*8d741a5dSApple OSS Distributions 		*dev_p = 0;
178*8d741a5dSApple OSS Distributions 	}
179*8d741a5dSApple OSS Distributions 
180*8d741a5dSApple OSS Distributions 	if (!path) {
181*8d741a5dSApple OSS Distributions 		return kIOReturnBadArgument;
182*8d741a5dSApple OSS Distributions 	}
183*8d741a5dSApple OSS Distributions 	if (!devname) {
184*8d741a5dSApple OSS Distributions 		return kIOReturnBadArgument;
185*8d741a5dSApple OSS Distributions 	}
186*8d741a5dSApple OSS Distributions 	if (!dev_p) {
187*8d741a5dSApple OSS Distributions 		return kIOReturnBadArgument;
188*8d741a5dSApple OSS Distributions 	}
189*8d741a5dSApple OSS Distributions 
190*8d741a5dSApple OSS Distributions 	controller = di_load_controller();
191*8d741a5dSApple OSS Distributions 	if (!controller) {
192*8d741a5dSApple OSS Distributions 		res = kIOReturnNotFound;
193*8d741a5dSApple OSS Distributions 		goto NoIOHDIXController;
194*8d741a5dSApple OSS Distributions 	}
195*8d741a5dSApple OSS Distributions 
196*8d741a5dSApple OSS Distributions 	// okay create path object
197*8d741a5dSApple OSS Distributions 	pathString = OSString::withCString(path);
198*8d741a5dSApple OSS Distributions 	if (!pathString) {
199*8d741a5dSApple OSS Distributions 		res = kIOReturnNoMemory;
200*8d741a5dSApple OSS Distributions 		goto CannotCreatePathOSString;
201*8d741a5dSApple OSS Distributions 	}
202*8d741a5dSApple OSS Distributions 
203*8d741a5dSApple OSS Distributions 	/*
204*8d741a5dSApple OSS Distributions 	 * This is a bit racy, as two concurrent attached could have
205*8d741a5dSApple OSS Distributions 	 * different properties. However, since we query the result and dev
206*8d741a5dSApple OSS Distributions 	 * below locklessly, the existing code is already racy, so we
207*8d741a5dSApple OSS Distributions 	 * keep the status quo.
208*8d741a5dSApple OSS Distributions 	 */
209*8d741a5dSApple OSS Distributions 	res = di_add_properties(controller, removable);
210*8d741a5dSApple OSS Distributions 	if (res) {
211*8d741a5dSApple OSS Distributions 		goto error_add_properties;
212*8d741a5dSApple OSS Distributions 	}
213*8d741a5dSApple OSS Distributions 
214*8d741a5dSApple OSS Distributions 	// do it
215*8d741a5dSApple OSS Distributions 	if (!controller->setProperty(kDIRootImageKey, pathString)) {
216*8d741a5dSApple OSS Distributions 		IOLog("IOHDIXController::setProperty(%s, %s) failed.\n", kDIRootImageKey, pathString->getCStringNoCopy());
217*8d741a5dSApple OSS Distributions 	}
218*8d741a5dSApple OSS Distributions 
219*8d741a5dSApple OSS Distributions 	myResult = OSDynamicCast(OSNumber, controller->getProperty(kDIRootImageResultKey));
220*8d741a5dSApple OSS Distributions 	res = kIOReturnError;
221*8d741a5dSApple OSS Distributions 	if (myResult) {
222*8d741a5dSApple OSS Distributions 		res = myResult->unsigned32BitValue();
223*8d741a5dSApple OSS Distributions 	}
224*8d741a5dSApple OSS Distributions 
225*8d741a5dSApple OSS Distributions 	if (res) {
226*8d741a5dSApple OSS Distributions 		IOLog("%s is 0x%08X/%d\n", kDIRootImageResultKey, res, res);
227*8d741a5dSApple OSS Distributions 		goto di_root_image_FAILED;
228*8d741a5dSApple OSS Distributions 	}
229*8d741a5dSApple OSS Distributions 
230*8d741a5dSApple OSS Distributions 	// success - grab
231*8d741a5dSApple OSS Distributions 	myDevT = OSDynamicCast(OSNumber, controller->getProperty(kDIRootImageDevTKey));
232*8d741a5dSApple OSS Distributions 	if (myDevT) {
233*8d741a5dSApple OSS Distributions 		*dev_p = myDevT->unsigned32BitValue();
234*8d741a5dSApple OSS Distributions 	} else {
235*8d741a5dSApple OSS Distributions 		IOLog("could not get %s\n", kDIRootImageDevTKey);
236*8d741a5dSApple OSS Distributions 		res = kIOReturnError;
237*8d741a5dSApple OSS Distributions 		goto di_root_image_FAILED;
238*8d741a5dSApple OSS Distributions 	}
239*8d741a5dSApple OSS Distributions 
240*8d741a5dSApple OSS Distributions 	myDevName = OSDynamicCast(OSString, controller->getProperty(kDIRootImageDevNameKey));
241*8d741a5dSApple OSS Distributions 	if (myDevName) {
242*8d741a5dSApple OSS Distributions 		strlcpy(devname, myDevName->getCStringNoCopy(), devsz);
243*8d741a5dSApple OSS Distributions 	} else {
244*8d741a5dSApple OSS Distributions 		IOLog("could not get %s\n", kDIRootImageDevNameKey);
245*8d741a5dSApple OSS Distributions 		res = kIOReturnError;
246*8d741a5dSApple OSS Distributions 		goto di_root_image_FAILED;
247*8d741a5dSApple OSS Distributions 	}
248*8d741a5dSApple OSS Distributions 
249*8d741a5dSApple OSS Distributions 	/*
250*8d741a5dSApple OSS Distributions 	 * NOTE: The attached disk image may trigger IOKit matching. At the very least, an IOMedia
251*8d741a5dSApple OSS Distributions 	 * must claim it.  More complex scenarios might include a GPT containing a partition mapping
252*8d741a5dSApple OSS Distributions 	 * to an APFS container, both of which need to probe and claim their respective media devices.
253*8d741a5dSApple OSS Distributions 	 *
254*8d741a5dSApple OSS Distributions 	 * After the attach is complete, we should quiesce the disk image controller before returning
255*8d741a5dSApple OSS Distributions 	 * from this function successfully.  If we failed to quiesce, then we should treat it as a hard
256*8d741a5dSApple OSS Distributions 	 * failure, to make it more obvious to triage.
257*8d741a5dSApple OSS Distributions 	 */
258*8d741a5dSApple OSS Distributions 	res = controller->waitQuiet((NSEC_PER_SEC * kDIMatchQuiesceTimeout));
259*8d741a5dSApple OSS Distributions 	if (res) {
260*8d741a5dSApple OSS Distributions 		IOLog("failed to quiesce attached disk image (%s)! \n", devname);
261*8d741a5dSApple OSS Distributions 		goto di_root_image_FAILED;
262*8d741a5dSApple OSS Distributions 	}
263*8d741a5dSApple OSS Distributions 
264*8d741a5dSApple OSS Distributions di_root_image_FAILED:
265*8d741a5dSApple OSS Distributions CannotCreatePathOSString:
266*8d741a5dSApple OSS Distributions NoIOHDIXController:
267*8d741a5dSApple OSS Distributions error_add_properties:
268*8d741a5dSApple OSS Distributions 
269*8d741a5dSApple OSS Distributions 	// clean up memory allocations
270*8d741a5dSApple OSS Distributions 	if (pathString) {
271*8d741a5dSApple OSS Distributions 		pathString->release();
272*8d741a5dSApple OSS Distributions 	}
273*8d741a5dSApple OSS Distributions 	if (controller) {
274*8d741a5dSApple OSS Distributions 		controller->release();
275*8d741a5dSApple OSS Distributions 	}
276*8d741a5dSApple OSS Distributions 
277*8d741a5dSApple OSS Distributions 	return res;
278*8d741a5dSApple OSS Distributions }
279*8d741a5dSApple OSS Distributions 
280*8d741a5dSApple OSS Distributions /*
281*8d741a5dSApple OSS Distributions  *       Name:		di_root_image
282*8d741a5dSApple OSS Distributions  *       Function:	mount the disk image returning the dev node
283*8d741a5dSApple OSS Distributions  *       Parameters:	path	->		path/url to disk image
284*8d741a5dSApple OSS Distributions  *                               devname	<-		dev node used to set the rootdevice global variable
285*8d741a5dSApple OSS Distributions  *                               dev_p	<-		device number generated from major/minor numbers
286*8d741a5dSApple OSS Distributions  *       Comments:
287*8d741a5dSApple OSS Distributions  *       This is an exported function. Changing this will break API.
288*8d741a5dSApple OSS Distributions  */
289*8d741a5dSApple OSS Distributions int
di_root_image(const char * path,char * devname,size_t devsz,dev_t * dev_p)290*8d741a5dSApple OSS Distributions di_root_image(const char *path, char *devname, size_t devsz, dev_t *dev_p)
291*8d741a5dSApple OSS Distributions {
292*8d741a5dSApple OSS Distributions 	return di_root_image_ext(path, devname, devsz, dev_p, false);
293*8d741a5dSApple OSS Distributions }
294*8d741a5dSApple OSS Distributions 
295*8d741a5dSApple OSS Distributions int
di_root_ramfile_buf(void * buf,size_t bufsz,char * devname,size_t devsz,dev_t * dev_p)296*8d741a5dSApple OSS Distributions di_root_ramfile_buf(void *buf, size_t bufsz, char *devname, size_t devsz, dev_t *dev_p)
297*8d741a5dSApple OSS Distributions {
298*8d741a5dSApple OSS Distributions 	IOReturn res = 0;
299*8d741a5dSApple OSS Distributions 	IOService *controller = NULL;
300*8d741a5dSApple OSS Distributions 	OSNumber *myResult = NULL;
301*8d741a5dSApple OSS Distributions 	OSString *myDevName = NULL;
302*8d741a5dSApple OSS Distributions 	OSNumber *myDevT = NULL;
303*8d741a5dSApple OSS Distributions 	IOMemoryDescriptor *mem = NULL;
304*8d741a5dSApple OSS Distributions 
305*8d741a5dSApple OSS Distributions 	/* Use kIOMemoryAutoPrepare and wire down the buffer so readBytes() will work. */
306*8d741a5dSApple OSS Distributions 	mem = IOMemoryDescriptor::withAddressRange(
307*8d741a5dSApple OSS Distributions 		(mach_vm_address_t)buf, (mach_vm_size_t)bufsz,
308*8d741a5dSApple OSS Distributions 		kIODirectionOut | kIOMemoryAutoPrepare, kernel_task);
309*8d741a5dSApple OSS Distributions 	if (!mem) {
310*8d741a5dSApple OSS Distributions 		res = kIOReturnNoMemory;
311*8d741a5dSApple OSS Distributions 		goto out;
312*8d741a5dSApple OSS Distributions 	}
313*8d741a5dSApple OSS Distributions 
314*8d741a5dSApple OSS Distributions 	controller = di_load_controller();
315*8d741a5dSApple OSS Distributions 	if (controller) {
316*8d741a5dSApple OSS Distributions 		/* attach the image */
317*8d741a5dSApple OSS Distributions 		controller->setProperty(kDIRootRamFileKey, mem);
318*8d741a5dSApple OSS Distributions 		controller->release();
319*8d741a5dSApple OSS Distributions 	} else {
320*8d741a5dSApple OSS Distributions 		res = kIOReturnNotFound;
321*8d741a5dSApple OSS Distributions 		goto out;
322*8d741a5dSApple OSS Distributions 	}
323*8d741a5dSApple OSS Distributions 
324*8d741a5dSApple OSS Distributions 	myResult = OSDynamicCast(OSNumber, controller->getProperty(kDIRootImageResultKey));
325*8d741a5dSApple OSS Distributions 	res = kIOReturnError;
326*8d741a5dSApple OSS Distributions 	if (myResult) {
327*8d741a5dSApple OSS Distributions 		res = myResult->unsigned32BitValue();
328*8d741a5dSApple OSS Distributions 	}
329*8d741a5dSApple OSS Distributions 
330*8d741a5dSApple OSS Distributions 	if (res) {
331*8d741a5dSApple OSS Distributions 		IOLog("%s is 0x%08X/%d\n", kDIRootImageResultKey, res, res);
332*8d741a5dSApple OSS Distributions 		goto out;
333*8d741a5dSApple OSS Distributions 	}
334*8d741a5dSApple OSS Distributions 
335*8d741a5dSApple OSS Distributions 	myDevT = OSDynamicCast(OSNumber, controller->getProperty(kDIRootImageDevTKey));
336*8d741a5dSApple OSS Distributions 	if (myDevT) {
337*8d741a5dSApple OSS Distributions 		*dev_p = myDevT->unsigned32BitValue();
338*8d741a5dSApple OSS Distributions 	} else {
339*8d741a5dSApple OSS Distributions 		IOLog("could not get %s\n", kDIRootImageDevTKey);
340*8d741a5dSApple OSS Distributions 		res = kIOReturnError;
341*8d741a5dSApple OSS Distributions 		goto out;
342*8d741a5dSApple OSS Distributions 	}
343*8d741a5dSApple OSS Distributions 
344*8d741a5dSApple OSS Distributions 	myDevName = OSDynamicCast(OSString, controller->getProperty(kDIRootImageDevNameKey));
345*8d741a5dSApple OSS Distributions 	if (myDevName) {
346*8d741a5dSApple OSS Distributions 		strlcpy(devname, myDevName->getCStringNoCopy(), devsz);
347*8d741a5dSApple OSS Distributions 	} else {
348*8d741a5dSApple OSS Distributions 		IOLog("could not get %s\n", kDIRootImageDevNameKey);
349*8d741a5dSApple OSS Distributions 		res = kIOReturnError;
350*8d741a5dSApple OSS Distributions 		goto out;
351*8d741a5dSApple OSS Distributions 	}
352*8d741a5dSApple OSS Distributions 
353*8d741a5dSApple OSS Distributions out:
354*8d741a5dSApple OSS Distributions 	OSSafeReleaseNULL(mem);
355*8d741a5dSApple OSS Distributions 	return res;
356*8d741a5dSApple OSS Distributions }
357*8d741a5dSApple OSS Distributions 
358*8d741a5dSApple OSS Distributions void
di_root_ramfile(IORegistryEntry * entry)359*8d741a5dSApple OSS Distributions di_root_ramfile( IORegistryEntry * entry )
360*8d741a5dSApple OSS Distributions {
361*8d741a5dSApple OSS Distributions 	OSData *                data;
362*8d741a5dSApple OSS Distributions 	IOMemoryDescriptor *    mem;
363*8d741a5dSApple OSS Distributions 	uint64_t                dmgSize;
364*8d741a5dSApple OSS Distributions 	uint64_t                remain, length;
365*8d741a5dSApple OSS Distributions 	OSData *                extentData = NULL;
366*8d741a5dSApple OSS Distributions 	IOAddressRange *        extentList;
367*8d741a5dSApple OSS Distributions 	uint64_t                extentSize;
368*8d741a5dSApple OSS Distributions 	uint32_t                extentCount;
369*8d741a5dSApple OSS Distributions 
370*8d741a5dSApple OSS Distributions 	do {
371*8d741a5dSApple OSS Distributions 		data = OSDynamicCast(OSData, entry->getProperty("boot-ramdmg-size"));
372*8d741a5dSApple OSS Distributions 		if (!data || (data->getLength() != sizeof(uint64_t))) {
373*8d741a5dSApple OSS Distributions 			break; // bad disk image size
374*8d741a5dSApple OSS Distributions 		}
375*8d741a5dSApple OSS Distributions 		dmgSize = *(uint64_t *) data->getBytesNoCopy();
376*8d741a5dSApple OSS Distributions 		if (!dmgSize) {
377*8d741a5dSApple OSS Distributions 			break;
378*8d741a5dSApple OSS Distributions 		}
379*8d741a5dSApple OSS Distributions 
380*8d741a5dSApple OSS Distributions 		data = OSDynamicCast(OSData, entry->getProperty("boot-ramdmg-extents"));
381*8d741a5dSApple OSS Distributions 		if (!data || (data->getLength() == 0) ||
382*8d741a5dSApple OSS Distributions 		    ((data->getLength() & (sizeof(IOAddressRange) - 1)) != 0)) {
383*8d741a5dSApple OSS Distributions 			break; // bad extents
384*8d741a5dSApple OSS Distributions 		}
385*8d741a5dSApple OSS Distributions 		// make modifications to local copy
386*8d741a5dSApple OSS Distributions 		extentData  = OSData::withData(data);
387*8d741a5dSApple OSS Distributions 		assert(extentData);
388*8d741a5dSApple OSS Distributions 
389*8d741a5dSApple OSS Distributions 		/* BEGIN IGNORE CODESTYLE */
390*8d741a5dSApple OSS Distributions 		__typed_allocators_ignore_push
391*8d741a5dSApple OSS Distributions 		extentList  = (IOAddressRange *) extentData->getBytesNoCopy();
392*8d741a5dSApple OSS Distributions 		__typed_allocators_ignore_pop
393*8d741a5dSApple OSS Distributions 		/* END IGNORE CODESTYLE */
394*8d741a5dSApple OSS Distributions 		extentCount = extentData->getLength() / sizeof(IOAddressRange);
395*8d741a5dSApple OSS Distributions 		extentSize  = 0;
396*8d741a5dSApple OSS Distributions 		remain = dmgSize;
397*8d741a5dSApple OSS Distributions 
398*8d741a5dSApple OSS Distributions 		// truncate extent length to enclosing disk image
399*8d741a5dSApple OSS Distributions 		for (uint32_t i = 0; i < extentCount; i++) {
400*8d741a5dSApple OSS Distributions 			length = extentList[i].length;
401*8d741a5dSApple OSS Distributions 			if (!length) {
402*8d741a5dSApple OSS Distributions 				break;
403*8d741a5dSApple OSS Distributions 			}
404*8d741a5dSApple OSS Distributions 
405*8d741a5dSApple OSS Distributions 			extentSize += length;
406*8d741a5dSApple OSS Distributions 			if (length >= remain) {
407*8d741a5dSApple OSS Distributions 				extentList[i].length = remain;
408*8d741a5dSApple OSS Distributions 				extentCount = i + 1;
409*8d741a5dSApple OSS Distributions 				break;
410*8d741a5dSApple OSS Distributions 			}
411*8d741a5dSApple OSS Distributions 			remain -= length;
412*8d741a5dSApple OSS Distributions 		}
413*8d741a5dSApple OSS Distributions 		if (extentSize < dmgSize) {
414*8d741a5dSApple OSS Distributions 			break; // not enough extent bytes for enclosing disk image
415*8d741a5dSApple OSS Distributions 		}
416*8d741a5dSApple OSS Distributions 		mem = IOMemoryDescriptor::withAddressRanges(
417*8d741a5dSApple OSS Distributions 			extentList, extentCount,
418*8d741a5dSApple OSS Distributions 			kIODirectionOut | kIOMemoryMapperNone, NULL);
419*8d741a5dSApple OSS Distributions 
420*8d741a5dSApple OSS Distributions 		if (mem) {
421*8d741a5dSApple OSS Distributions 			IOService * controller = di_load_controller();
422*8d741a5dSApple OSS Distributions 			if (controller) {
423*8d741a5dSApple OSS Distributions 				controller->setProperty(kDIRootRamFileKey, mem);
424*8d741a5dSApple OSS Distributions 				controller->release();
425*8d741a5dSApple OSS Distributions 			}
426*8d741a5dSApple OSS Distributions 			mem->release();
427*8d741a5dSApple OSS Distributions 		}
428*8d741a5dSApple OSS Distributions 	} while (false);
429*8d741a5dSApple OSS Distributions 
430*8d741a5dSApple OSS Distributions 	if (extentData) {
431*8d741a5dSApple OSS Distributions 		extentData->release();
432*8d741a5dSApple OSS Distributions 	}
433*8d741a5dSApple OSS Distributions }
434*8d741a5dSApple OSS Distributions };
435