VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/ldr/ldrFile.cpp@ 25524

Last change on this file since 25524 was 21337, checked in by vboxsync, 16 years ago

IPRT,HostDrv,AddDrv: Export public IPRT symbols for the linux kernel (pain).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 10.3 KB
Line 
1/* $Id: ldrFile.cpp 21337 2009-07-07 14:58:27Z vboxsync $ */
2/** @file
3 * IPRT - Binary Image Loader, The File Oriented Parts.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 *
26 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
27 * Clara, CA 95054 USA or visit http://www.sun.com if you need
28 * additional information or have any questions.
29 */
30
31
32/*******************************************************************************
33* Header Files *
34*******************************************************************************/
35#define LOG_GROUP RTLOGGROUP_LDR
36#include <iprt/ldr.h>
37#include "internal/iprt.h"
38
39#include <iprt/alloc.h>
40#include <iprt/file.h>
41#include <iprt/assert.h>
42#include <iprt/log.h>
43#include <iprt/err.h>
44#include <iprt/string.h>
45#include "internal/ldr.h"
46#include "internal/ldrMZ.h"
47
48
49/*******************************************************************************
50* Structures and Typedefs *
51*******************************************************************************/
52/**
53 * File Reader instance.
54 * This provides raw image bits from a file.
55 */
56typedef struct RTLDRREADERFILE
57{
58 /** The core. */
59 RTLDRREADER Core;
60 /** The file. */
61 RTFILE File;
62 /** The file size. */
63 RTFOFF cbFile;
64 /** The current offset. */
65 RTFOFF off;
66 /** Number of users or the mapping. */
67 RTUINT cMappings;
68 /** Pointer to the in memory mapping. */
69 void *pvMapping;
70 /** The filename (variable size). */
71 char szFilename[1];
72} RTLDRREADERFILE, *PRTLDRREADERFILE;
73
74
75/** @copydoc RTLDRREADER::pfnRead */
76static DECLCALLBACK(int) rtldrFileRead(PRTLDRREADER pReader, void *pvBuf, size_t cb, RTFOFF off)
77{
78 PRTLDRREADERFILE pFileReader = (PRTLDRREADERFILE)pReader;
79
80 /*
81 * Seek.
82 */
83 if (pFileReader->off != off)
84 {
85 int rc = RTFileSeek(pFileReader->File, off, RTFILE_SEEK_BEGIN, NULL);
86 if (RT_FAILURE(rc))
87 {
88 pFileReader->off = -1;
89 return rc;
90 }
91 pFileReader->off = off;
92 }
93
94 /*
95 * Read.
96 */
97 int rc = RTFileRead(pFileReader->File, pvBuf, cb, NULL);
98 if (RT_SUCCESS(rc))
99 pFileReader->off += cb;
100 else
101 pFileReader->off = -1;
102 return rc;
103}
104
105
106/** @copydoc RTLDRREADER::pfnTell */
107static DECLCALLBACK(RTFOFF) rtldrFileTell(PRTLDRREADER pReader)
108{
109 PRTLDRREADERFILE pFileReader = (PRTLDRREADERFILE)pReader;
110 return pFileReader->off;
111}
112
113
114/** @copydoc RTLDRREADER::pfnSize */
115static DECLCALLBACK(RTFOFF) rtldrFileSize(PRTLDRREADER pReader)
116{
117 PRTLDRREADERFILE pFileReader = (PRTLDRREADERFILE)pReader;
118 return pFileReader->cbFile;
119}
120
121
122/** @copydoc RTLDRREADER::pfnLogName */
123static DECLCALLBACK(const char *) rtldrFileLogName(PRTLDRREADER pReader)
124{
125 PRTLDRREADERFILE pFileReader = (PRTLDRREADERFILE)pReader;
126 return pFileReader->szFilename;
127}
128
129
130/** @copydoc RTLDRREADER::pfnMap */
131static DECLCALLBACK(int) rtldrFileMap(PRTLDRREADER pReader, const void **ppvBits)
132{
133 PRTLDRREADERFILE pFileReader = (PRTLDRREADERFILE)pReader;
134
135 /*
136 * Already mapped?
137 */
138 if (pFileReader->pvMapping)
139 {
140 pFileReader->cMappings++;
141 *ppvBits = pFileReader->pvMapping;
142 return VINF_SUCCESS;
143 }
144
145 /*
146 * Allocate memory.
147 */
148 size_t cb = (size_t)pFileReader->cbFile;
149 if ((RTFOFF)cb != pFileReader->cbFile)
150 return VERR_IMAGE_TOO_BIG;
151 pFileReader->pvMapping = RTMemAlloc(cb);
152 if (!pFileReader->pvMapping)
153 return VERR_NO_MEMORY;
154 int rc = rtldrFileRead(pReader, pFileReader->pvMapping, cb, 0);
155 if (RT_SUCCESS(rc))
156 {
157 pFileReader->cMappings = 1;
158 *ppvBits = pFileReader->pvMapping;
159 }
160 else
161 {
162 RTMemFree(pFileReader->pvMapping);
163 pFileReader->pvMapping = NULL;
164 }
165
166 return rc;
167}
168
169
170/** @copydoc RTLDRREADER::pfnUnmap */
171static DECLCALLBACK(int) rtldrFileUnmap(PRTLDRREADER pReader, const void *pvBits)
172{
173 PRTLDRREADERFILE pFileReader = (PRTLDRREADERFILE)pReader;
174 AssertReturn(pFileReader->cMappings > 0, VERR_INVALID_PARAMETER);
175
176 if (!--pFileReader->cMappings)
177 {
178 RTMemFree(pFileReader->pvMapping);
179 pFileReader->pvMapping = NULL;
180 }
181
182 return VINF_SUCCESS;
183}
184
185
186/** @copydoc RTLDRREADER::pfnDestroy */
187static DECLCALLBACK(int) rtldrFileDestroy(PRTLDRREADER pReader)
188{
189 int rc = VINF_SUCCESS;
190 PRTLDRREADERFILE pFileReader = (PRTLDRREADERFILE)pReader;
191 if (pFileReader->File != NIL_RTFILE)
192 {
193 rc = RTFileClose(pFileReader->File);
194 AssertRC(rc);
195 pFileReader->File = NIL_RTFILE;
196 }
197 RTMemFree(pFileReader);
198 return rc;
199}
200
201
202/**
203 * Opens a loader file reader.
204 *
205 * @returns iprt status code.
206 * @param ppReader Where to store the reader instance on success.
207 * @param pszFilename The file to open.
208 */
209static int rtldrFileCreate(PRTLDRREADER *ppReader, const char *pszFilename)
210{
211 size_t cchFilename = strlen(pszFilename);
212 int rc = VERR_NO_MEMORY;
213 PRTLDRREADERFILE pFileReader = (PRTLDRREADERFILE)RTMemAlloc(sizeof(*pFileReader) + cchFilename);
214 if (pFileReader)
215 {
216 memcpy(pFileReader->szFilename, pszFilename, cchFilename + 1);
217 rc = RTFileOpen(&pFileReader->File, pszFilename, RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_WRITE);
218 if (RT_SUCCESS(rc))
219 {
220 rc = RTFileGetSize(pFileReader->File, (uint64_t *)&pFileReader->cbFile);
221 if (RT_SUCCESS(rc))
222 {
223 pFileReader->Core.pfnRead = rtldrFileRead;
224 pFileReader->Core.pfnTell = rtldrFileTell;
225 pFileReader->Core.pfnSize = rtldrFileSize;
226 pFileReader->Core.pfnLogName = rtldrFileLogName;
227 pFileReader->Core.pfnMap = rtldrFileMap;
228 pFileReader->Core.pfnUnmap = rtldrFileUnmap;
229 pFileReader->Core.pfnDestroy = rtldrFileDestroy;
230 pFileReader->off = 0;
231 pFileReader->cMappings = 0;
232 pFileReader->pvMapping = NULL;
233 *ppReader = &pFileReader->Core;
234 return VINF_SUCCESS;
235 }
236 RTFileClose(pFileReader->File);
237 }
238 RTMemFree(pFileReader);
239 }
240 *ppReader = NULL;
241 return rc;
242}
243
244
245/**
246 * Open a binary image file, extended version.
247 *
248 * @returns iprt status code.
249 * @param pszFilename Image filename.
250 * @param fFlags Reserved, MBZ.
251 * @param enmArch CPU architecture specifier for the image to be loaded.
252 * @param phLdrMod Where to store the handle to the loader module.
253 */
254RTDECL(int) RTLdrOpen(const char *pszFilename, uint32_t fFlags, RTLDRARCH enmArch, PRTLDRMOD phLdrMod)
255{
256 LogFlow(("RTLdrOpen: pszFilename=%p:{%s} fFlags=%#x enmArch=%d phLdrMod=%p\n",
257 pszFilename, pszFilename, fFlags, enmArch, phLdrMod));
258 AssertMsgReturn(!fFlags, ("%#x\n", fFlags), VERR_INVALID_PARAMETER);
259 AssertMsgReturn(enmArch > RTLDRARCH_INVALID && enmArch < RTLDRARCH_END, ("%d\n", enmArch), VERR_INVALID_PARAMETER);
260
261 /*
262 * Resolve RTLDRARCH_HOST.
263 */
264 if (enmArch == RTLDRARCH_HOST)
265#if defined(RT_ARCH_AMD64)
266 enmArch = RTLDRARCH_AMD64;
267#elif defined(RT_ARCH_X86)
268 enmArch = RTLDRARCH_X86_32;
269#else
270 enmArch = RTLDRARCH_WHATEVER;
271#endif
272
273 /*
274 * Create file reader & invoke worker which identifies and calls the image interpreter.
275 */
276 PRTLDRREADER pReader;
277 int rc = rtldrFileCreate(&pReader, pszFilename);
278 if (RT_SUCCESS(rc))
279 {
280 rc = rtldrOpenWithReader(pReader, fFlags, enmArch, phLdrMod);
281 if (RT_SUCCESS(rc))
282 {
283 LogFlow(("RTLdrOpen: return %Rrc *phLdrMod\n", rc, *phLdrMod));
284 return rc;
285 }
286 pReader->pfnDestroy(pReader);
287 }
288 *phLdrMod = NIL_RTLDRMOD;
289 LogFlow(("RTLdrOpen: return %Rrc\n", rc));
290 return rc;
291}
292RT_EXPORT_SYMBOL(RTLdrOpen);
293
294
295/**
296 * Opens a binary image file using kLdr.
297 *
298 * @returns iprt status code.
299 * @param pszFilename Image filename.
300 * @param fFlags Reserved, MBZ.
301 * @param enmArch CPU architecture specifier for the image to be loaded.
302 * @param phLdrMod Where to store the handle to the loaded module.
303 * @remark Primarily for testing the loader.
304 */
305RTDECL(int) RTLdrOpenkLdr(const char *pszFilename, uint32_t fFlags, RTLDRARCH enmArch, PRTLDRMOD phLdrMod)
306{
307#ifdef LDR_WITH_KLDR
308 LogFlow(("RTLdrOpenkLdr: pszFilename=%p:{%s} fFlags=%#x enmArch=%d phLdrMod=%p\n",
309 pszFilename, pszFilename, fFlags, enmArch, phLdrMod));
310
311 /*
312 * Resolve RTLDRARCH_HOST.
313 */
314 if (enmArch == RTLDRARCH_HOST)
315# if defined(RT_ARCH_AMD64)
316 enmArch = RTLDRARCH_AMD64;
317# elif defined(RT_ARCH_X86)
318 enmArch = RTLDRARCH_X86_32;
319# else
320 enmArch = RTLDRARCH_WHATEVER;
321# endif
322
323 /*
324 * Create file reader & invoke worker which identifies and calls the image interpreter.
325 */
326 PRTLDRREADER pReader;
327 int rc = rtldrFileCreate(&pReader, pszFilename);
328 if (RT_SUCCESS(rc))
329 {
330 rc = rtldrkLdrOpen(pReader, fFlags, enmArch, phLdrMod);
331 if (RT_SUCCESS(rc))
332 {
333 LogFlow(("RTLdrOpenkLdr: return %Rrc *phLdrMod\n", rc, *phLdrMod));
334 return rc;
335 }
336 pReader->pfnDestroy(pReader);
337 }
338 *phLdrMod = NIL_RTLDRMOD;
339 LogFlow(("RTLdrOpenkLdr: return %Rrc\n", rc));
340 return rc;
341
342#else
343 return RTLdrOpen(pszFilename, fFlags, enmArch, phLdrMod);
344#endif
345}
346RT_EXPORT_SYMBOL(RTLdrOpenkLdr);
347
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette