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