This adds two functions IsValidTimeZone() and IsValidDaylight() to check the time zone and daylight value from EFI time. These functions are retrieved from the RealTimeClockRuntimeDxe module as they reduce duplicated code in RTC modules. Cc: Leif Lindholm <leif@nuviainc.com> Cc: Ard Biesheuvel <ard.biesheuvel@arm.com> Signed-off-by: Nhi Pham <nhi@os.amperecomputing.com> Reviewed-by: Leif Lindholm <leif@nuviainc.com>
		
			
				
	
	
		
			290 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			290 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/** @file
 | 
						|
*
 | 
						|
*  Copyright (c) 2016, Hisilicon Limited. All rights reserved.
 | 
						|
*  Copyright (c) 2016-2019, Linaro Limited. All rights reserved.
 | 
						|
*  Copyright (c) 2021, Ampere Computing LLC. All rights reserved.
 | 
						|
*
 | 
						|
*  SPDX-License-Identifier: BSD-2-Clause-Patent
 | 
						|
*
 | 
						|
**/
 | 
						|
 | 
						|
#include <Uefi/UefiBaseType.h>
 | 
						|
#include <Uefi/UefiSpec.h>
 | 
						|
#include <Library/DebugLib.h>
 | 
						|
#include <Library/TimeBaseLib.h>
 | 
						|
 | 
						|
/**
 | 
						|
  Converts Epoch seconds (elapsed since 1970 JANUARY 01, 00:00:00 UTC) to EFI_TIME.
 | 
						|
 | 
						|
  @param  EpochSeconds   Epoch seconds.
 | 
						|
  @param  Time           The time converted to UEFI format.
 | 
						|
 | 
						|
**/
 | 
						|
VOID
 | 
						|
EFIAPI
 | 
						|
EpochToEfiTime (
 | 
						|
  IN  UINTN     EpochSeconds,
 | 
						|
  OUT EFI_TIME  *Time
 | 
						|
  )
 | 
						|
{
 | 
						|
  UINTN         a;
 | 
						|
  UINTN         b;
 | 
						|
  UINTN         c;
 | 
						|
  UINTN         d;
 | 
						|
  UINTN         g;
 | 
						|
  UINTN         j;
 | 
						|
  UINTN         m;
 | 
						|
  UINTN         y;
 | 
						|
  UINTN         da;
 | 
						|
  UINTN         db;
 | 
						|
  UINTN         dc;
 | 
						|
  UINTN         dg;
 | 
						|
  UINTN         hh;
 | 
						|
  UINTN         mm;
 | 
						|
  UINTN         ss;
 | 
						|
  UINTN         J;
 | 
						|
 | 
						|
  J  = (EpochSeconds / 86400) + 2440588;
 | 
						|
  j  = J + 32044;
 | 
						|
  g  = j / 146097;
 | 
						|
  dg = j % 146097;
 | 
						|
  c  = (((dg / 36524) + 1) * 3) / 4;
 | 
						|
  dc = dg - (c * 36524);
 | 
						|
  b  = dc / 1461;
 | 
						|
  db = dc % 1461;
 | 
						|
  a  = (((db / 365) + 1) * 3) / 4;
 | 
						|
  da = db - (a * 365);
 | 
						|
  y  = (g * 400) + (c * 100) + (b * 4) + a;
 | 
						|
  m  = (((da * 5) + 308) / 153) - 2;
 | 
						|
  d  = da - (((m + 4) * 153) / 5) + 122;
 | 
						|
 | 
						|
  Time->Year  = (UINT16)(y - 4800 + ((m + 2) / 12));
 | 
						|
  Time->Month = ((m + 2) % 12) + 1;
 | 
						|
  Time->Day   = (UINT8)(d + 1);
 | 
						|
 | 
						|
  ss = EpochSeconds % 60;
 | 
						|
  a  = (EpochSeconds - ss) / 60;
 | 
						|
  mm = a % 60;
 | 
						|
  b = (a - mm) / 60;
 | 
						|
  hh = b % 24;
 | 
						|
 | 
						|
  Time->Hour        = (UINT8)hh;
 | 
						|
  Time->Minute      = (UINT8)mm;
 | 
						|
  Time->Second      = (UINT8)ss;
 | 
						|
  Time->Nanosecond  = 0;
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
  Calculate Epoch days.
 | 
						|
 | 
						|
  @param    Time  The UEFI time to be calculated.
 | 
						|
 | 
						|
  @return   Number of days.
 | 
						|
 | 
						|
**/
 | 
						|
UINTN
 | 
						|
EFIAPI
 | 
						|
EfiGetEpochDays (
 | 
						|
  IN  EFI_TIME  *Time
 | 
						|
  )
 | 
						|
{
 | 
						|
  UINTN a;
 | 
						|
  UINTN y;
 | 
						|
  UINTN m;
 | 
						|
  UINTN JulianDate;  // Absolute Julian Date representation of the supplied Time
 | 
						|
  UINTN EpochDays;   // Number of days elapsed since EPOCH_JULIAN_DAY
 | 
						|
 | 
						|
  a = (14 - Time->Month) / 12 ;
 | 
						|
  y = Time->Year + 4800 - a;
 | 
						|
  m = Time->Month + (12*a) - 3;
 | 
						|
 | 
						|
  JulianDate = Time->Day + ((153*m + 2)/5) + (365*y) + (y/4) - (y/100) + (y/400) - 32045;
 | 
						|
 | 
						|
  ASSERT (JulianDate >= EPOCH_JULIAN_DATE);
 | 
						|
  EpochDays = JulianDate - EPOCH_JULIAN_DATE;
 | 
						|
 | 
						|
  return EpochDays;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
  Converts EFI_TIME to Epoch seconds (elapsed since 1970 JANUARY 01, 00:00:00 UTC).
 | 
						|
 | 
						|
  @param    Time  The UEFI time to be converted.
 | 
						|
 | 
						|
  @return   Number of seconds.
 | 
						|
 | 
						|
**/
 | 
						|
UINTN
 | 
						|
EFIAPI
 | 
						|
EfiTimeToEpoch (
 | 
						|
  IN  EFI_TIME  *Time
 | 
						|
  )
 | 
						|
{
 | 
						|
  UINTN EpochDays;   // Number of days elapsed since EPOCH_JULIAN_DAY
 | 
						|
  UINTN EpochSeconds;
 | 
						|
 | 
						|
  EpochDays = EfiGetEpochDays (Time);
 | 
						|
 | 
						|
  EpochSeconds = (EpochDays * SEC_PER_DAY) + ((UINTN)Time->Hour * SEC_PER_HOUR) + (Time->Minute * SEC_PER_MIN) + Time->Second;
 | 
						|
 | 
						|
  return EpochSeconds;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
  Get the day of the week from the UEFI time.
 | 
						|
 | 
						|
  @param    Time  The UEFI time to be calculated.
 | 
						|
 | 
						|
  @return   The day of the week: Sunday=0, Monday=1, ... Saturday=6
 | 
						|
 | 
						|
**/
 | 
						|
UINTN
 | 
						|
EfiTimeToWday (
 | 
						|
  IN  EFI_TIME  *Time
 | 
						|
  )
 | 
						|
{
 | 
						|
  UINTN EpochDays;   // Number of days elapsed since EPOCH_JULIAN_DAY
 | 
						|
 | 
						|
  EpochDays = EfiGetEpochDays (Time);
 | 
						|
 | 
						|
  // 4=1/1/1970 was a Thursday
 | 
						|
 | 
						|
  return (EpochDays + 4) % 7;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
  Check if it is a leap year.
 | 
						|
 | 
						|
  @param    Time  The UEFI time to be checked.
 | 
						|
 | 
						|
  @retval   TRUE  It is a leap year.
 | 
						|
  @retval   FALSE It is NOT a leap year.
 | 
						|
 | 
						|
**/
 | 
						|
BOOLEAN
 | 
						|
EFIAPI
 | 
						|
IsLeapYear (
 | 
						|
  IN EFI_TIME   *Time
 | 
						|
  )
 | 
						|
{
 | 
						|
  if (Time->Year % 4 == 0) {
 | 
						|
    if (Time->Year % 100 == 0) {
 | 
						|
      if (Time->Year % 400 == 0) {
 | 
						|
        return TRUE;
 | 
						|
      } else {
 | 
						|
        return FALSE;
 | 
						|
      }
 | 
						|
    } else {
 | 
						|
      return TRUE;
 | 
						|
    }
 | 
						|
  } else {
 | 
						|
    return FALSE;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
  Check if the day in the UEFI time is valid.
 | 
						|
 | 
						|
  @param    Time    The UEFI time to be checked.
 | 
						|
 | 
						|
  @retval   TRUE    Valid.
 | 
						|
  @retval   FALSE   Invalid.
 | 
						|
 | 
						|
**/
 | 
						|
BOOLEAN
 | 
						|
EFIAPI
 | 
						|
IsDayValid (
 | 
						|
  IN  EFI_TIME  *Time
 | 
						|
  )
 | 
						|
{
 | 
						|
  STATIC CONST INTN DayOfMonth[12] = { 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
 | 
						|
 | 
						|
  if (Time->Day < 1 ||
 | 
						|
      Time->Day > DayOfMonth[Time->Month - 1] ||
 | 
						|
      (Time->Month == 2 && (!IsLeapYear (Time) && Time->Day > 28))
 | 
						|
     ) {
 | 
						|
    return FALSE;
 | 
						|
  }
 | 
						|
 | 
						|
  return TRUE;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
  Check if the time zone is valid.
 | 
						|
  Valid values are between -1440 and 1440 or 2047 (EFI_UNSPECIFIED_TIMEZONE).
 | 
						|
 | 
						|
  @param    TimeZone    The time zone to be checked.
 | 
						|
 | 
						|
  @retval   TRUE    Valid.
 | 
						|
  @retval   FALSE   Invalid.
 | 
						|
 | 
						|
**/
 | 
						|
BOOLEAN
 | 
						|
EFIAPI
 | 
						|
IsValidTimeZone (
 | 
						|
  IN  INT16  TimeZone
 | 
						|
  )
 | 
						|
{
 | 
						|
  return TimeZone == EFI_UNSPECIFIED_TIMEZONE ||
 | 
						|
         (TimeZone >= -1440 && TimeZone <= 1440);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
  Check if the daylight is valid.
 | 
						|
  Valid values are:
 | 
						|
    0 : Time is not affected.
 | 
						|
    1 : Time is affected, and has not been adjusted for daylight savings.
 | 
						|
    3 : Time is affected, and has been adjusted for daylight savings.
 | 
						|
  All other values are invalid.
 | 
						|
 | 
						|
  @param    Daylight    The daylight to be checked.
 | 
						|
 | 
						|
  @retval   TRUE    Valid.
 | 
						|
  @retval   FALSE   Invalid.
 | 
						|
 | 
						|
**/
 | 
						|
BOOLEAN
 | 
						|
EFIAPI
 | 
						|
IsValidDaylight (
 | 
						|
  IN  INT8  Daylight
 | 
						|
  )
 | 
						|
{
 | 
						|
  return Daylight == 0 ||
 | 
						|
         Daylight == EFI_TIME_ADJUST_DAYLIGHT ||
 | 
						|
         Daylight == (EFI_TIME_ADJUST_DAYLIGHT | EFI_TIME_IN_DAYLIGHT);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
  Check if the UEFI time is valid.
 | 
						|
 | 
						|
  @param    Time    The UEFI time to be checked.
 | 
						|
 | 
						|
  @retval   TRUE    Valid.
 | 
						|
  @retval   FALSE   Invalid.
 | 
						|
 | 
						|
**/
 | 
						|
BOOLEAN
 | 
						|
EFIAPI
 | 
						|
IsTimeValid (
 | 
						|
  IN EFI_TIME *Time
 | 
						|
  )
 | 
						|
{
 | 
						|
  // Check the input parameters are within the range specified by UEFI
 | 
						|
  if ((Time->Year  < 2000)              ||
 | 
						|
     (Time->Year   > 2099)              ||
 | 
						|
     (Time->Month  < 1   )              ||
 | 
						|
     (Time->Month  > 12  )              ||
 | 
						|
     (!IsDayValid (Time) )              ||
 | 
						|
     (Time->Hour   > 23  )              ||
 | 
						|
     (Time->Minute > 59  )              ||
 | 
						|
     (Time->Second > 59  )              ||
 | 
						|
     (Time->Nanosecond > 999999999)     ||
 | 
						|
     (!IsValidTimeZone(Time->TimeZone)) ||
 | 
						|
     (!IsValidDaylight(Time->Daylight))) {
 | 
						|
    return FALSE;
 | 
						|
  }
 | 
						|
 | 
						|
  return TRUE;
 | 
						|
}
 |