| 
									
										
										
										
											2004-02-16 22:12:40 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * QEMU low level functions | 
					
						
							| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2004-02-16 22:12:40 +00:00
										 |  |  |  * Copyright (c) 2003 Fabrice Bellard | 
					
						
							| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2004-02-16 22:12:40 +00:00
										 |  |  |  * Permission is hereby granted, free of charge, to any person obtaining a copy | 
					
						
							|  |  |  |  * of this software and associated documentation files (the "Software"), to deal | 
					
						
							|  |  |  |  * in the Software without restriction, including without limitation the rights | 
					
						
							|  |  |  |  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | 
					
						
							|  |  |  |  * copies of the Software, and to permit persons to whom the Software is | 
					
						
							|  |  |  |  * furnished to do so, subject to the following conditions: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The above copyright notice and this permission notice shall be included in | 
					
						
							|  |  |  |  * all copies or substantial portions of the Software. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
					
						
							|  |  |  |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
					
						
							|  |  |  |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL | 
					
						
							|  |  |  |  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | 
					
						
							|  |  |  |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | 
					
						
							|  |  |  |  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | 
					
						
							|  |  |  |  * THE SOFTWARE. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <stdarg.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <errno.h>
 | 
					
						
							|  |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2007-03-25 21:33:06 +00:00
										 |  |  | #include <fcntl.h>
 | 
					
						
							| 
									
										
										
										
											2007-01-17 23:31:19 +00:00
										 |  |  | #ifdef HOST_SOLARIS
 | 
					
						
							|  |  |  | #include <sys/types.h>
 | 
					
						
							|  |  |  | #include <sys/statvfs.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-02-16 22:12:40 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "cpu.h"
 | 
					
						
							| 
									
										
										
										
											2006-05-22 14:10:48 +00:00
										 |  |  | #if defined(USE_KQEMU)
 | 
					
						
							|  |  |  | #include "vl.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-02-16 22:12:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-04-17 18:33:47 +00:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | #include <windows.h>
 | 
					
						
							|  |  |  | #elif defined(_BSD)
 | 
					
						
							| 
									
										
										
										
											2005-02-21 20:10:36 +00:00
										 |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  | #include <malloc.h>
 | 
					
						
							| 
									
										
										
										
											2005-02-21 20:10:36 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-02-16 22:12:40 +00:00
										 |  |  | void *get_mmap_addr(unsigned long size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void qemu_free(void *ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     free(ptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void *qemu_malloc(size_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return malloc(size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-04-17 18:33:47 +00:00
										 |  |  | #if defined(_WIN32)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void *qemu_vmalloc(size_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* FIXME: this is not exactly optimal solution since VirtualAlloc
 | 
					
						
							|  |  |  |        has 64Kb granularity, but at least it guarantees us that the | 
					
						
							|  |  |  |        memory is page aligned. */ | 
					
						
							|  |  |  |     return VirtualAlloc(NULL, size, MEM_COMMIT, PAGE_READWRITE); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void qemu_vfree(void *ptr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     VirtualFree(ptr, 0, MEM_RELEASE); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-05-22 14:10:48 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined(USE_KQEMU)
 | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-04-23 20:44:25 +00:00
										 |  |  | #include <sys/vfs.h>
 | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  | #include <sys/mman.h>
 | 
					
						
							|  |  |  | #include <fcntl.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-05-22 14:10:48 +00:00
										 |  |  | void *kqemu_vmalloc(size_t size) | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     static int phys_ram_fd = -1; | 
					
						
							|  |  |  |     static int phys_ram_size = 0; | 
					
						
							|  |  |  |     const char *tmpdir; | 
					
						
							|  |  |  |     char phys_ram_file[1024]; | 
					
						
							|  |  |  |     void *ptr; | 
					
						
							| 
									
										
										
										
											2007-01-17 23:31:19 +00:00
										 |  |  | #ifdef HOST_SOLARIS
 | 
					
						
							|  |  |  |     struct statvfs stfs; | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2005-04-23 20:44:25 +00:00
										 |  |  |     struct statfs stfs; | 
					
						
							| 
									
										
										
										
											2007-01-17 23:31:19 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (phys_ram_fd < 0) { | 
					
						
							|  |  |  |         tmpdir = getenv("QEMU_TMPDIR"); | 
					
						
							|  |  |  |         if (!tmpdir) | 
					
						
							| 
									
										
										
										
											2007-01-17 23:31:19 +00:00
										 |  |  | #ifdef HOST_SOLARIS
 | 
					
						
							|  |  |  |             tmpdir = "/tmp"; | 
					
						
							|  |  |  |         if (statvfs(tmpdir, &stfs) == 0) { | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  |             tmpdir = "/dev/shm"; | 
					
						
							| 
									
										
										
										
											2005-04-23 20:44:25 +00:00
										 |  |  |         if (statfs(tmpdir, &stfs) == 0) { | 
					
						
							| 
									
										
										
										
											2007-01-17 23:31:19 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-04-23 20:44:25 +00:00
										 |  |  |             int64_t free_space; | 
					
						
							|  |  |  |             int ram_mb; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             extern int ram_size; | 
					
						
							|  |  |  |             free_space = (int64_t)stfs.f_bavail * stfs.f_bsize; | 
					
						
							|  |  |  |             if ((ram_size + 8192 * 1024) >= free_space) { | 
					
						
							|  |  |  |                 ram_mb = (ram_size / (1024 * 1024)); | 
					
						
							| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  |                 fprintf(stderr, | 
					
						
							| 
									
										
										
										
											2005-04-23 20:44:25 +00:00
										 |  |  |                         "You do not have enough space in '%s' for the %d MB of QEMU virtual RAM.\n", | 
					
						
							|  |  |  |                         tmpdir, ram_mb); | 
					
						
							|  |  |  |                 if (strcmp(tmpdir, "/dev/shm") == 0) { | 
					
						
							|  |  |  |                     fprintf(stderr, "To have more space available provided you have enough RAM and swap, do as root:\n" | 
					
						
							|  |  |  |                             "umount /dev/shm\n" | 
					
						
							|  |  |  |                             "mount -t tmpfs -o size=%dm none /dev/shm\n", | 
					
						
							|  |  |  |                             ram_mb + 16); | 
					
						
							|  |  |  |                 } else { | 
					
						
							| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  |                     fprintf(stderr, | 
					
						
							| 
									
										
										
										
											2005-04-23 20:44:25 +00:00
										 |  |  |                             "Use the '-m' option of QEMU to diminish the amount of virtual RAM or use the\n" | 
					
						
							|  |  |  |                             "QEMU_TMPDIR environment variable to set another directory where the QEMU\n" | 
					
						
							|  |  |  |                             "temporary RAM file will be opened.\n"); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2006-05-22 14:10:48 +00:00
										 |  |  |                 fprintf(stderr, "Or disable the accelerator module with -no-kqemu\n"); | 
					
						
							| 
									
										
										
										
											2005-04-23 20:44:25 +00:00
										 |  |  |                 exit(1); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  |         snprintf(phys_ram_file, sizeof(phys_ram_file), "%s/qemuXXXXXX", | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  |                  tmpdir); | 
					
						
							| 
									
										
										
										
											2006-08-17 16:19:07 +00:00
										 |  |  |         phys_ram_fd = mkstemp(phys_ram_file); | 
					
						
							|  |  |  |         if (phys_ram_fd < 0) { | 
					
						
							| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  |             fprintf(stderr, | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  |                     "warning: could not create temporary file in '%s'.\n" | 
					
						
							|  |  |  |                     "Use QEMU_TMPDIR to select a directory in a tmpfs filesystem.\n" | 
					
						
							|  |  |  |                     "Using '/tmp' as fallback.\n", | 
					
						
							|  |  |  |                     tmpdir); | 
					
						
							| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  |             snprintf(phys_ram_file, sizeof(phys_ram_file), "%s/qemuXXXXXX", | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  |                      "/tmp"); | 
					
						
							| 
									
										
										
										
											2006-08-17 16:19:07 +00:00
										 |  |  |             phys_ram_fd = mkstemp(phys_ram_file); | 
					
						
							|  |  |  |             if (phys_ram_fd < 0) { | 
					
						
							| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  |                 fprintf(stderr, "Could not create temporary memory file '%s'\n", | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  |                         phys_ram_file); | 
					
						
							|  |  |  |                 exit(1); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         unlink(phys_ram_file); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     size = (size + 4095) & ~4095; | 
					
						
							|  |  |  |     ftruncate(phys_ram_fd, phys_ram_size + size); | 
					
						
							| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  |     ptr = mmap(NULL, | 
					
						
							|  |  |  |                size, | 
					
						
							|  |  |  |                PROT_WRITE | PROT_READ, MAP_SHARED, | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  |                phys_ram_fd, phys_ram_size); | 
					
						
							|  |  |  |     if (ptr == MAP_FAILED) { | 
					
						
							|  |  |  |         fprintf(stderr, "Could not map physical memory\n"); | 
					
						
							|  |  |  |         exit(1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     phys_ram_size += size; | 
					
						
							|  |  |  |     return ptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-05-22 14:10:48 +00:00
										 |  |  | void kqemu_vfree(void *ptr) | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     /* may be useful some day, but currently we do not need to free */ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-05-22 14:10:48 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* alloc shared memory pages */ | 
					
						
							|  |  |  | void *qemu_vmalloc(size_t size) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2006-05-22 14:10:48 +00:00
										 |  |  | #if defined(USE_KQEMU)
 | 
					
						
							|  |  |  |     if (kqemu_allowed) | 
					
						
							|  |  |  |         return kqemu_vmalloc(size); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  | #ifdef _BSD
 | 
					
						
							|  |  |  |     return valloc(size); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     return memalign(4096, size); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void qemu_vfree(void *ptr) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2006-05-22 14:10:48 +00:00
										 |  |  | #if defined(USE_KQEMU)
 | 
					
						
							|  |  |  |     if (kqemu_allowed) | 
					
						
							|  |  |  |         kqemu_vfree(ptr); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-02-10 21:59:25 +00:00
										 |  |  |     free(ptr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-03-14 21:41:12 +00:00
										 |  |  | void *qemu_mallocz(size_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     void *ptr; | 
					
						
							|  |  |  |     ptr = qemu_malloc(size); | 
					
						
							|  |  |  |     if (!ptr) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     memset(ptr, 0, size); | 
					
						
							|  |  |  |     return ptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-14 17:21:57 +00:00
										 |  |  | char *qemu_strdup(const char *str) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     char *ptr; | 
					
						
							|  |  |  |     ptr = qemu_malloc(strlen(str) + 1); | 
					
						
							|  |  |  |     if (!ptr) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     strcpy(ptr, str); | 
					
						
							|  |  |  |     return ptr; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2007-03-25 21:33:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | int qemu_create_pidfile(const char *filename) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     char buffer[128]; | 
					
						
							|  |  |  |     int len; | 
					
						
							|  |  |  | #ifndef _WIN32
 | 
					
						
							|  |  |  |     int fd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fd = open(filename, O_RDWR | O_CREAT, 0600); | 
					
						
							|  |  |  |     if (fd == -1) | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (lockf(fd, F_TLOCK, 0) == -1) | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     len = snprintf(buffer, sizeof(buffer), "%ld\n", (long)getpid()); | 
					
						
							|  |  |  |     if (write(fd, buffer, len) != len) | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     HANDLE file; | 
					
						
							|  |  |  |     DWORD flags; | 
					
						
							|  |  |  |     OVERLAPPED overlap; | 
					
						
							|  |  |  |     BOOL ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Open for writing with no sharing. */ | 
					
						
							| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  |     file = CreateFile(filename, GENERIC_WRITE, 0, NULL, | 
					
						
							| 
									
										
										
										
											2007-03-25 21:33:06 +00:00
										 |  |  | 		      OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (file == INVALID_HANDLE_VALUE) | 
					
						
							|  |  |  |       return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     flags = LOCKFILE_EXCLUSIVE_LOCK | LOCKFILE_FAIL_IMMEDIATELY; | 
					
						
							|  |  |  |     overlap.hEvent = 0; | 
					
						
							|  |  |  |     /* Lock 1 byte. */ | 
					
						
							|  |  |  |     ret = LockFileEx(file, flags, 0, 0, 1, &overlap); | 
					
						
							|  |  |  |     if (ret == 0) | 
					
						
							|  |  |  |       return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Write PID to file. */ | 
					
						
							|  |  |  |     len = snprintf(buffer, sizeof(buffer), "%ld\n", (long)getpid()); | 
					
						
							| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  |     ret = WriteFileEx(file, (LPCVOID)buffer, (DWORD)len, | 
					
						
							| 
									
										
										
										
											2007-03-25 21:33:06 +00:00
										 |  |  | 		      &overlap, NULL); | 
					
						
							|  |  |  |     if (ret == 0) | 
					
						
							|  |  |  |       return -1; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2007-06-07 23:09:47 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Offset between 1/1/1601 and 1/1/1970 in 100 nanosec units */ | 
					
						
							|  |  |  | #define _W32_FT_OFFSET (116444736000000000ULL)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int qemu_gettimeofday(qemu_timeval *tp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   union { | 
					
						
							|  |  |  |     unsigned long long ns100; /*time since 1 Jan 1601 in 100ns units */ | 
					
						
							|  |  |  |     FILETIME ft; | 
					
						
							|  |  |  |   }  _now; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if(tp) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       GetSystemTimeAsFileTime (&_now.ft); | 
					
						
							|  |  |  |       tp->tv_usec=(long)((_now.ns100 / 10ULL) % 1000000ULL ); | 
					
						
							|  |  |  |       tp->tv_sec= (long)((_now.ns100 - _W32_FT_OFFSET) / 10000000ULL); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   /* Always return 0 as per Open Group Base Specifications Issue 6.
 | 
					
						
							|  |  |  |      Do not set errno on error.  */ | 
					
						
							|  |  |  |   return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif /* _WIN32 */
 |