git-svn-id: https://edk2.svn.sourceforge.net/svnroot/edk2/trunk/edk2@11414 6f19259b-4bc3-4df7-8a09-765794883524
		
			
				
	
	
		
			344 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			344 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*++
 | |
| 
 | |
| Copyright (c) 2004 - 2009, Intel Corporation. All rights reserved.<BR>
 | |
| Portions copyright (c) 2008 - 2010, Apple Inc. All rights reserved.<BR>
 | |
| 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"
 | |
| 
 | |
| #if defined(__APPLE__) || defined(MDE_CPU_X64)
 | |
| #include "Gasket.h"
 | |
| #endif
 | |
| 
 | |
| int settimer_initialized;
 | |
| struct timeval settimer_timeval;
 | |
| void (*settimer_callback)(UINT64 delta);
 | |
| 
 | |
| BOOLEAN gEmulatorInterruptEnabled = FALSE;
 | |
| 
 | |
| 
 | |
| 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) {
 | |
| #if defined(__APPLE__) || defined(MDE_CPU_X64)
 | |
|    ReverseGasketUint64 (settimer_callback, delta);
 | |
| #else
 | |
|     (*settimer_callback)(delta);
 | |
| #endif
 | |
|   }
 | |
| }
 | |
| 
 | |
| 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);
 | |
|     gEmulatorInterruptEnabled = TRUE;
 | |
|     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
 | |
| UnixEnableInterrupt (void)
 | |
| {
 | |
|   sigset_t  sigset;
 | |
| 
 | |
|   gEmulatorInterruptEnabled = TRUE;
 | |
|   // Since SetTimer() uses SIGALRM we emulate turning on and off interrupts 
 | |
|   // by enabling/disabling SIGALRM.
 | |
|   sigemptyset (&sigset);
 | |
|   sigaddset (&sigset, SIGALRM);
 | |
|   sigprocmask (SIG_UNBLOCK, &sigset, NULL);
 | |
| }
 | |
| 
 | |
| 
 | |
| void
 | |
| UnixDisableInterrupt (void)
 | |
| {
 | |
|   sigset_t  sigset;
 | |
| 
 | |
|   // Since SetTimer() uses SIGALRM we emulate turning on and off interrupts 
 | |
|   // by enabling/disabling SIGALRM.
 | |
|   sigemptyset (&sigset);
 | |
|   sigaddset (&sigset, SIGALRM);
 | |
|   sigprocmask (SIG_BLOCK, &sigset, NULL);
 | |
|   gEmulatorInterruptEnabled = FALSE;
 | |
| }
 | |
| 
 | |
| 
 | |
| BOOLEAN
 | |
| UnixInterruptEanbled (void)
 | |
| {
 | |
|   return gEmulatorInterruptEnabled;
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| void
 | |
| msSleep (unsigned long Milliseconds)
 | |
| {
 | |
|   struct timespec rq, rm;
 | |
|   struct timeval  start, end;
 | |
|   unsigned long  MicroSec;
 | |
|   
 | |
|   rq.tv_sec = Milliseconds / 1000;
 | |
|   rq.tv_nsec = (Milliseconds % 1000) * 1000000;
 | |
| 
 | |
|   //
 | |
|   // nanosleep gets interrupted by our timer tic. 
 | |
|   // we need to track wall clock time or we will stall for way too long
 | |
|   //
 | |
|   gettimeofday (&start, NULL);
 | |
|   end.tv_sec  = start.tv_sec + rq.tv_sec;
 | |
|   MicroSec = (start.tv_usec + rq.tv_nsec/1000);
 | |
|   end.tv_usec = MicroSec % 1000000;
 | |
|   if (MicroSec > 1000000) {
 | |
|     end.tv_sec++;
 | |
|   }
 | |
| 
 | |
|   while (nanosleep (&rq, &rm) == -1) {
 | |
|     if (errno != EINTR) {
 | |
|       break;
 | |
|     }
 | |
|     gettimeofday (&start, NULL);
 | |
|     if (start.tv_sec > end.tv_sec) {
 | |
|       break;
 | |
|     } if ((start.tv_sec == end.tv_sec) && (start.tv_usec > end.tv_usec)) {
 | |
|       break;
 | |
|     }
 | |
|     rq = rm;
 | |
|   } 
 | |
| }
 | |
| 
 | |
| 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 = GetTimeZone ();
 | |
|   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,
 | |
| #if defined(__APPLE__) || defined(MDE_CPU_X64)
 | |
| //
 | |
| // 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,
 | |
|   Gasketpoll,
 | |
|   Gasketread,
 | |
|   Gasketwrite,
 | |
|   Gasketgetenv,
 | |
|   Gasketopen,
 | |
|   Gasketlseek,
 | |
|   Gasketftruncate,
 | |
|   Gasketclose,
 | |
|   Gasketmkdir,
 | |
|   Gasketrmdir,
 | |
|   Gasketunlink,
 | |
|   GasketGetErrno,
 | |
|   Gasketopendir,
 | |
|   Gasketrewinddir,
 | |
|   Gasketreaddir,
 | |
|   Gasketclosedir,
 | |
|   Gasketstat,
 | |
|   Gasketstatfs,
 | |
|   Gasketrename,
 | |
|   Gasketmktime,
 | |
|   Gasketfsync,
 | |
|   Gasketchmod,
 | |
|   Gasketutime,
 | |
|   Gaskettcflush,
 | |
|   GasketUgaCreate,
 | |
|   Gasketperror,
 | |
|   Gasketioctl,
 | |
|   Gasketfcntl,
 | |
|   Gasketcfsetispeed,
 | |
|   Gasketcfsetospeed,
 | |
|   Gaskettcgetattr,
 | |
|   Gaskettcsetattr,
 | |
|   GasketUnixPeCoffGetEntryPoint,                
 | |
|   GasketUnixPeCoffRelocateImageExtraAction,     
 | |
|   GasketUnixPeCoffUnloadImageExtraAction,  
 | |
|   
 | |
|   GasketUnixEnableInterrupt,
 | |
|   GasketUnixDisableInterrupt,
 | |
| 
 | |
|   Gasketgetifaddrs,
 | |
|   Gasketfreeifaddrs,
 | |
|   Gasketsocket,
 | |
| 
 | |
| #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,
 | |
|   SecPeCoffGetEntryPoint,
 | |
|   SecPeCoffRelocateImageExtraAction,
 | |
|   SecPeCoffLoaderUnloadImageExtraAction,
 | |
|   UnixEnableInterrupt,
 | |
|   UnixDisableInterrupt,
 | |
|   getifaddrs,
 | |
|   freeifaddrs,
 | |
|   socket
 | |
| #endif
 | |
| };
 | |
| 
 | |
| 
 | |
| EFI_UNIX_THUNK_PROTOCOL *gUnix = &mUnixThunkTable;
 |