mirror of
				https://github.com/acidanthera/audk.git
				synced 2025-10-26 08:43:46 +01:00 
			
		
		
		
	git-svn-id: https://edk2.svn.sourceforge.net/svnroot/edk2/trunk/edk2@9519 6f19259b-4bc3-4df7-8a09-765794883524
		
			
				
	
	
		
			269 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			269 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*++
 | |
| 
 | |
| Copyright (c) 2004 - 2009, Intel Corporation                                                         
 | |
| Portions copyright (c) 2008-2009 Apple Inc. All rights reserved.
 | |
| All rights reserved. This program and the accompanying materials                          
 | |
| are licensed and made available under the terms and conditions of the BSD License         
 | |
| which accompanies this distribution.  The full text of the license may be found at        
 | |
| http://opensource.org/licenses/bsd-license.php                                            
 | |
|                                                                                           
 | |
| THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,                     
 | |
| WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.             
 | |
| 
 | |
| Module Name:
 | |
| 
 | |
|   UnixThunk.c
 | |
| 
 | |
| Abstract:
 | |
| 
 | |
|   Since the SEC is the only program in our emulation we 
 | |
|   must use a Tiano mechanism to export APIs to other modules.
 | |
|   This is the role of the EFI_UNIX_THUNK_PROTOCOL.
 | |
| 
 | |
|   The mUnixThunkTable exists so that a change to EFI_UNIX_THUNK_PROTOCOL
 | |
|   will cause an error in initializing the array if all the member functions
 | |
|   are not added. It looks like adding a element to end and not initializing
 | |
|   it may cause the table to be initaliized with the members at the end being
 | |
|   set to zero. This is bad as jumping to zero will crash.
 | |
|   
 | |
| 
 | |
|   gUnix is a a public exported global that contains the initialized
 | |
|   data.
 | |
| 
 | |
| --*/
 | |
| 
 | |
| #include "SecMain.h"
 | |
| #include "Uefi.h"
 | |
| #include "Library/UnixLib.h"
 | |
| 
 | |
| #ifdef __APPLE__
 | |
| #include "Gasket.h"
 | |
| #endif
 | |
| 
 | |
| int settimer_initialized;
 | |
| struct timeval settimer_timeval;
 | |
| void (*settimer_callback)(UINT64 delta);
 | |
| 
 | |
| void
 | |
| settimer_handler (int sig)
 | |
| {
 | |
|   struct timeval timeval;
 | |
|   UINT64 delta;
 | |
| 
 | |
|   gettimeofday (&timeval, NULL);
 | |
|   delta = ((UINT64)timeval.tv_sec * 1000) + (timeval.tv_usec / 1000)
 | |
|     - ((UINT64)settimer_timeval.tv_sec * 1000) 
 | |
|     - (settimer_timeval.tv_usec / 1000);
 | |
|   settimer_timeval = timeval;
 | |
|   if (settimer_callback)
 | |
|     (*settimer_callback)(delta);
 | |
| }
 | |
| 
 | |
| VOID
 | |
| SetTimer (UINT64 PeriodMs, VOID (*CallBack)(UINT64 DeltaMs))
 | |
| {
 | |
|   struct itimerval timerval;
 | |
|   UINT32 remainder;
 | |
| 
 | |
|   if (!settimer_initialized) {
 | |
|     struct sigaction act;
 | |
| 
 | |
|     settimer_initialized = 1;
 | |
|     act.sa_handler = settimer_handler;
 | |
|     act.sa_flags = 0;
 | |
|     sigemptyset (&act.sa_mask);
 | |
|     if (sigaction (SIGALRM, &act, NULL) != 0) {
 | |
|       printf ("SetTimer: sigaction error %s\n", strerror (errno));
 | |
|     }
 | |
|     if (gettimeofday (&settimer_timeval, NULL) != 0) {
 | |
|       printf ("SetTimer: gettimeofday error %s\n", strerror (errno));
 | |
|     }
 | |
|   }
 | |
|   timerval.it_value.tv_sec = DivU64x32(PeriodMs, 1000);
 | |
|   DivU64x32Remainder(PeriodMs, 1000, &remainder);
 | |
|   timerval.it_value.tv_usec = remainder * 1000;
 | |
|   timerval.it_value.tv_sec = DivU64x32(PeriodMs, 1000);
 | |
|   timerval.it_interval = timerval.it_value;
 | |
|   
 | |
|   if (setitimer (ITIMER_REAL, &timerval, NULL) != 0) {
 | |
|     printf ("SetTimer: setitimer error %s\n", strerror (errno));
 | |
|   }
 | |
|   settimer_callback = CallBack;
 | |
| }
 | |
| 
 | |
| void
 | |
| msSleep (unsigned long Milliseconds)
 | |
| {
 | |
|   struct timespec ts;
 | |
| 
 | |
|   ts.tv_sec = Milliseconds / 1000;
 | |
|   ts.tv_nsec = (Milliseconds % 1000) * 1000000;
 | |
| 
 | |
|   while (nanosleep (&ts, &ts) != 0 && errno == EINTR)
 | |
|     ;
 | |
| }
 | |
| 
 | |
| void
 | |
| GetLocalTime (EFI_TIME *Time)
 | |
| {
 | |
|   struct tm *tm;
 | |
|   time_t t;
 | |
| 
 | |
|   t = time (NULL);
 | |
|   tm = localtime (&t);
 | |
| 
 | |
|   Time->Year = 1900 + tm->tm_year;
 | |
|   Time->Month = tm->tm_mon + 1;
 | |
|   Time->Day = tm->tm_mday;
 | |
|   Time->Hour = tm->tm_hour;
 | |
|   Time->Minute = tm->tm_min;
 | |
|   Time->Second = tm->tm_sec;
 | |
|   Time->Nanosecond = 0;
 | |
|   Time->TimeZone = timezone;
 | |
|   Time->Daylight = (daylight ? EFI_TIME_ADJUST_DAYLIGHT : 0)
 | |
|     | (tm->tm_isdst > 0 ? EFI_TIME_IN_DAYLIGHT : 0);
 | |
| }
 | |
| 
 | |
| void
 | |
| TzSet (void)
 | |
| {
 | |
|   STATIC int done = 0;
 | |
|   if (!done) {
 | |
|     tzset ();
 | |
|     done = 1;
 | |
|   }
 | |
| }
 | |
| 
 | |
| long
 | |
| GetTimeZone(void)
 | |
| {
 | |
|   TzSet ();
 | |
|   return timezone;
 | |
| }
 | |
| 
 | |
| int
 | |
| GetDayLight(void)
 | |
| {
 | |
|   TzSet ();
 | |
|   return daylight;
 | |
| }
 | |
| 
 | |
| int
 | |
| GetErrno(void)
 | |
| {
 | |
|   return errno;
 | |
| }
 | |
| 
 | |
| 
 | |
| extern EFI_STATUS
 | |
| UgaCreate(struct _EFI_UNIX_UGA_IO_PROTOCOL **UgaIo, CONST CHAR16 *Title);
 | |
| 
 | |
| EFI_UNIX_THUNK_PROTOCOL mUnixThunkTable = {
 | |
|   EFI_UNIX_THUNK_PROTOCOL_SIGNATURE,
 | |
| #ifdef __APPLE__
 | |
| //
 | |
| // Mac OS X requires the stack to be 16-byte aligned for IA-32. So on an OS X build
 | |
| // we add an assembly wrapper that makes sure the stack ges aligned. 
 | |
| // This has the nice benfit of being able to run EFI ABI code, like the EFI shell
 | |
| // that is checked in to source control in the OS X version of the emulator
 | |
| //
 | |
|   GasketmsSleep, /* Sleep */
 | |
|   Gasketexit, /* Exit */
 | |
|   GasketSetTimer,
 | |
|   GasketGetLocalTime,
 | |
|   Gasketgmtime,
 | |
|   GasketGetTimeZone,
 | |
|   GasketGetDayLight,
 | |
|   (UnixPoll)Gasketpoll,
 | |
|   (UnixRead)Gasketread,
 | |
|   (UnixWrite)Gasketwrite,
 | |
|   Gasketgetenv,
 | |
|   (UnixOpen)Gasketopen,
 | |
|   (UnixSeek)Gasketlseek,
 | |
|   (UnixFtruncate)Gasketftruncate,
 | |
|   Gasketclose,
 | |
|   Gasketmkdir,
 | |
|   Gasketrmdir,
 | |
|   Gasketunlink,
 | |
|   GasketGetErrno,
 | |
|   Gasketopendir,
 | |
|   (UnixRewindDir)Gasketrewinddir,
 | |
|   Gasketreaddir,
 | |
|   Gasketclosedir,
 | |
|   Gasketstat,
 | |
|   Gasketstatfs,
 | |
|   Gasketrename,
 | |
|   Gasketmktime,
 | |
|   Gasketfsync,
 | |
|   Gasketchmod,
 | |
|   Gasketutime,
 | |
|   Gaskettcflush,
 | |
|   GasketUgaCreate,
 | |
|   Gasketperror,
 | |
|   Gasketioctl,
 | |
|   Gasketfcntl,
 | |
|   Gasketcfsetispeed,
 | |
|   Gasketcfsetospeed,
 | |
|   Gaskettcgetattr,
 | |
|   Gaskettcsetattr,
 | |
|   
 | |
|   dlopen,  // Update me with a gasket
 | |
|   dlerror, // Update me with a gasket
 | |
|   dlsym,   // Update me with a gasket
 | |
| 
 | |
|   SecPeCoffGetEntryPoint,                // Update me with a gasket
 | |
|   SecPeCoffRelocateImageExtraAction,     // Update me with a gasket
 | |
|   SecPeCoffLoaderUnloadImageExtraAction  // Update me with a gasket
 | |
| 
 | |
| #else
 | |
|   msSleep, /* Sleep */
 | |
|   exit, /* Exit */
 | |
|   SetTimer,
 | |
|   GetLocalTime,
 | |
|   gmtime,
 | |
|   GetTimeZone,
 | |
|   GetDayLight,
 | |
|   (UnixPoll)poll,
 | |
|   (UnixRead)read,
 | |
|   (UnixWrite)write,
 | |
|   getenv,
 | |
|   (UnixOpen)open,
 | |
|   (UnixSeek)lseek,
 | |
|   (UnixFtruncate)ftruncate,
 | |
|   close,
 | |
|   mkdir,
 | |
|   rmdir,
 | |
|   unlink,
 | |
|   GetErrno,
 | |
|   opendir,
 | |
|   rewinddir,
 | |
|   readdir,
 | |
|   closedir,
 | |
|   (UnixStat)stat,
 | |
|   statfs,
 | |
|   rename,
 | |
|   mktime,
 | |
|   fsync,
 | |
|   chmod,
 | |
|   utime,
 | |
|   tcflush,
 | |
|   UgaCreate,
 | |
|   perror,
 | |
|   ioctl,
 | |
|   fcntl,
 | |
|   cfsetispeed,
 | |
|   cfsetospeed,
 | |
|   tcgetattr,
 | |
|   tcsetattr,
 | |
|   dlopen,
 | |
|   dlerror,
 | |
|   dlsym,
 | |
|   SecPeCoffGetEntryPoint,
 | |
|   SecPeCoffRelocateImageExtraAction,
 | |
|   SecPeCoffLoaderUnloadImageExtraAction
 | |
| #endif
 | |
| };
 | |
| 
 | |
| 
 | |
| EFI_UNIX_THUNK_PROTOCOL *gUnix = &mUnixThunkTable;
 |