mirror of
https://github.com/CloverHackyColor/CloverBootloader.git
synced 2024-12-01 12:53:27 +01:00
594 lines
11 KiB
C
594 lines
11 KiB
C
/** @file
|
|
C Run-Time Libraries (CRT) Wrapper Implementation for OpenSSL-based
|
|
Cryptographic Library.
|
|
|
|
Copyright (c) 2009 - 2017, Intel Corporation. All rights reserved.<BR>
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
|
|
|
**/
|
|
|
|
#include <CrtLibSupport.h>
|
|
|
|
int errno = 0;
|
|
|
|
FILE *stderr = NULL;
|
|
FILE *stdin = NULL;
|
|
FILE *stdout = NULL;
|
|
|
|
typedef
|
|
int
|
|
(*SORT_COMPARE)(
|
|
IN VOID *Buffer1,
|
|
IN VOID *Buffer2
|
|
);
|
|
|
|
//
|
|
// Duplicated from EDKII BaseSortLib for qsort() wrapper
|
|
//
|
|
STATIC
|
|
VOID
|
|
QuickSortWorker (
|
|
IN OUT VOID *BufferToSort,
|
|
IN CONST UINTN Count,
|
|
IN CONST UINTN ElementSize,
|
|
IN SORT_COMPARE CompareFunction,
|
|
IN VOID *Buffer
|
|
)
|
|
{
|
|
VOID *Pivot;
|
|
UINTN LoopCount;
|
|
UINTN NextSwapLocation;
|
|
|
|
ASSERT (BufferToSort != NULL);
|
|
ASSERT (CompareFunction != NULL);
|
|
ASSERT (Buffer != NULL);
|
|
|
|
if ((Count < 2) || (ElementSize < 1)) {
|
|
return;
|
|
}
|
|
|
|
NextSwapLocation = 0;
|
|
|
|
//
|
|
// Pick a pivot (we choose last element)
|
|
//
|
|
Pivot = ((UINT8 *)BufferToSort + ((Count - 1) * ElementSize));
|
|
|
|
//
|
|
// Now get the pivot such that all on "left" are below it
|
|
// and everything "right" are above it
|
|
//
|
|
for (LoopCount = 0; LoopCount < Count - 1; LoopCount++) {
|
|
//
|
|
// If the element is less than the pivot
|
|
//
|
|
if (CompareFunction ((VOID *)((UINT8 *)BufferToSort + ((LoopCount) * ElementSize)), Pivot) <= 0) {
|
|
//
|
|
// Swap
|
|
//
|
|
CopyMem (Buffer, (UINT8 *)BufferToSort + (NextSwapLocation * ElementSize), ElementSize);
|
|
CopyMem ((UINT8 *)BufferToSort + (NextSwapLocation * ElementSize), (UINT8 *)BufferToSort + ((LoopCount) * ElementSize), ElementSize);
|
|
CopyMem ((UINT8 *)BufferToSort + ((LoopCount) * ElementSize), Buffer, ElementSize);
|
|
|
|
//
|
|
// Increment NextSwapLocation
|
|
//
|
|
NextSwapLocation++;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Swap pivot to its final position (NextSwapLocation)
|
|
//
|
|
CopyMem (Buffer, Pivot, ElementSize);
|
|
CopyMem (Pivot, (UINT8 *)BufferToSort + (NextSwapLocation * ElementSize), ElementSize);
|
|
CopyMem ((UINT8 *)BufferToSort + (NextSwapLocation * ElementSize), Buffer, ElementSize);
|
|
|
|
//
|
|
// Now recurse on 2 partial lists. Neither of these will have the 'pivot' element.
|
|
// IE list is sorted left half, pivot element, sorted right half...
|
|
//
|
|
QuickSortWorker (
|
|
BufferToSort,
|
|
NextSwapLocation,
|
|
ElementSize,
|
|
CompareFunction,
|
|
Buffer
|
|
);
|
|
|
|
QuickSortWorker (
|
|
(UINT8 *)BufferToSort + (NextSwapLocation + 1) * ElementSize,
|
|
Count - NextSwapLocation - 1,
|
|
ElementSize,
|
|
CompareFunction,
|
|
Buffer
|
|
);
|
|
|
|
return;
|
|
}
|
|
|
|
// ---------------------------------------------------------
|
|
// Standard C Run-time Library Interface Wrapper
|
|
// ---------------------------------------------------------
|
|
|
|
// Jief : functions we already have are commented out
|
|
//
|
|
// -- String Manipulation Routines --
|
|
//
|
|
|
|
//char *
|
|
//strchr (
|
|
// const char *str,
|
|
// int ch
|
|
// )
|
|
//{
|
|
// return ScanMem8 (str, AsciiStrSize (str), (UINT8)ch);
|
|
//}
|
|
//
|
|
///* Scan a string for the last occurrence of a character */
|
|
//char *
|
|
//strrchr (
|
|
// const char *str,
|
|
// int c
|
|
// )
|
|
//{
|
|
// char *save;
|
|
//
|
|
// for (save = NULL; ; ++str) {
|
|
// if (*str == c) {
|
|
// save = (char *)str;
|
|
// }
|
|
//
|
|
// if (*str == 0) {
|
|
// return (save);
|
|
// }
|
|
// }
|
|
//}
|
|
//
|
|
///* Compare first n bytes of string s1 with string s2, ignoring case */
|
|
//int
|
|
//strncasecmp (
|
|
// const char *s1,
|
|
// const char *s2,
|
|
// size_t n
|
|
// )
|
|
//{
|
|
// int Val;
|
|
//
|
|
// ASSERT (s1 != NULL);
|
|
// ASSERT (s2 != NULL);
|
|
//
|
|
// if (n != 0) {
|
|
// do {
|
|
// Val = tolower (*s1) - tolower (*s2);
|
|
// if (Val != 0) {
|
|
// return Val;
|
|
// }
|
|
//
|
|
// ++s1;
|
|
// ++s2;
|
|
// if (*s1 == '\0') {
|
|
// break;
|
|
// }
|
|
// } while (--n != 0);
|
|
// }
|
|
//
|
|
// return 0;
|
|
//}
|
|
|
|
/* Read formatted data from a string */
|
|
int
|
|
sscanf (
|
|
const char *buffer,
|
|
const char *format,
|
|
...
|
|
)
|
|
{
|
|
//
|
|
// Null sscanf() function implementation to satisfy the linker, since
|
|
// no direct functionality logic dependency in present UEFI cases.
|
|
//
|
|
return 0;
|
|
}
|
|
|
|
/* Maps errnum to an error-message string */
|
|
char *
|
|
strerror (
|
|
int errnum
|
|
)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
///* Computes the length of the maximum initial segment of the string pointed to by s1
|
|
// which consists entirely of characters from the string pointed to by s2. */
|
|
//size_t
|
|
//strspn (
|
|
// const char *s1,
|
|
// const char *s2
|
|
// )
|
|
//{
|
|
// UINT8 Map[32];
|
|
// UINT32 Index;
|
|
// size_t Count;
|
|
//
|
|
// for (Index = 0; Index < 32; Index++) {
|
|
// Map[Index] = 0;
|
|
// }
|
|
//
|
|
// while (*s2) {
|
|
// Map[*s2 >> 3] |= (1 << (*s2 & 7));
|
|
// s2++;
|
|
// }
|
|
//
|
|
// if (*s1) {
|
|
// Count = 0;
|
|
// while (Map[*s1 >> 3] & (1 << (*s1 & 7))) {
|
|
// Count++;
|
|
// s1++;
|
|
// }
|
|
//
|
|
// return Count;
|
|
// }
|
|
//
|
|
// return 0;
|
|
//}
|
|
//
|
|
///* Computes the length of the maximum initial segment of the string pointed to by s1
|
|
// which consists entirely of characters not from the string pointed to by s2. */
|
|
//size_t
|
|
//strcspn (
|
|
// const char *s1,
|
|
// const char *s2
|
|
// )
|
|
//{
|
|
// UINT8 Map[32];
|
|
// UINT32 Index;
|
|
// size_t Count;
|
|
//
|
|
// for (Index = 0; Index < 32; Index++) {
|
|
// Map[Index] = 0;
|
|
// }
|
|
//
|
|
// while (*s2) {
|
|
// Map[*s2 >> 3] |= (1 << (*s2 & 7));
|
|
// s2++;
|
|
// }
|
|
//
|
|
// Map[0] |= 1;
|
|
//
|
|
// Count = 0;
|
|
// while (!(Map[*s1 >> 3] & (1 << (*s1 & 7)))) {
|
|
// Count++;
|
|
// s1++;
|
|
// }
|
|
//
|
|
// return Count;
|
|
//}
|
|
//
|
|
//char *
|
|
//strcpy (
|
|
// char *strDest,
|
|
// const char *strSource
|
|
// )
|
|
//{
|
|
// AsciiStrCpyS (strDest, MAX_STRING_SIZE, strSource);
|
|
// return strDest;
|
|
//}
|
|
//
|
|
//int
|
|
//strcmp (
|
|
// const char *s1,
|
|
// const char *s2
|
|
// )
|
|
//{
|
|
// return (int)AsciiStrCmp (s1, s2);
|
|
//}
|
|
//
|
|
////
|
|
//// -- Character Classification Routines --
|
|
////
|
|
//
|
|
///* Determines if a particular character is a decimal-digit character */
|
|
//int
|
|
//isdigit (
|
|
// int c
|
|
// )
|
|
//{
|
|
// //
|
|
// // <digit> ::= [0-9]
|
|
// //
|
|
// return (('0' <= (c)) && ((c) <= '9'));
|
|
//}
|
|
//
|
|
///* Determine if an integer represents character that is a hex digit */
|
|
//int
|
|
//isxdigit (
|
|
// int c
|
|
// )
|
|
//{
|
|
// //
|
|
// // <hexdigit> ::= [0-9] | [a-f] | [A-F]
|
|
// //
|
|
// return ((('0' <= (c)) && ((c) <= '9')) ||
|
|
// (('a' <= (c)) && ((c) <= 'f')) ||
|
|
// (('A' <= (c)) && ((c) <= 'F')));
|
|
//}
|
|
|
|
/* Determines if a particular character represents a space character */
|
|
int
|
|
isspace (
|
|
int c
|
|
)
|
|
{
|
|
//
|
|
// <space> ::= [ ]
|
|
//
|
|
return ((c) == ' ');
|
|
}
|
|
|
|
///* Determine if a particular character is an alphanumeric character */
|
|
//int
|
|
//isalnum (
|
|
// int c
|
|
// )
|
|
//{
|
|
// //
|
|
// // <alnum> ::= [0-9] | [a-z] | [A-Z]
|
|
// //
|
|
// return ((('0' <= (c)) && ((c) <= '9')) ||
|
|
// (('a' <= (c)) && ((c) <= 'z')) ||
|
|
// (('A' <= (c)) && ((c) <= 'Z')));
|
|
//}
|
|
//
|
|
///* Determines if a particular character is in upper case */
|
|
//int
|
|
//isupper (
|
|
// int c
|
|
// )
|
|
//{
|
|
// //
|
|
// // <uppercase letter> := [A-Z]
|
|
// //
|
|
// return (('A' <= (c)) && ((c) <= 'Z'));
|
|
//}
|
|
|
|
//
|
|
// -- Data Conversion Routines --
|
|
//
|
|
|
|
/* Convert strings to a long-integer value */
|
|
long
|
|
strtol (
|
|
const char *nptr,
|
|
char **endptr,
|
|
int base
|
|
)
|
|
{
|
|
//
|
|
// Null strtol() function implementation to satisfy the linker, since there is
|
|
// no direct functionality logic dependency in present UEFI cases.
|
|
//
|
|
return 0;
|
|
}
|
|
|
|
/* Convert strings to an unsigned long-integer value */
|
|
unsigned long
|
|
strtoul (
|
|
const char *nptr,
|
|
char **endptr,
|
|
int base
|
|
)
|
|
{
|
|
//
|
|
// Null strtoul() function implementation to satisfy the linker, since there is
|
|
// no direct functionality logic dependency in present UEFI cases.
|
|
//
|
|
return 0;
|
|
}
|
|
|
|
///* Convert character to lowercase */
|
|
//int
|
|
//tolower (
|
|
// int c
|
|
// )
|
|
//{
|
|
// if (('A' <= (c)) && ((c) <= 'Z')) {
|
|
// return (c - ('A' - 'a'));
|
|
// }
|
|
//
|
|
// return (c);
|
|
//}
|
|
|
|
//
|
|
// -- Searching and Sorting Routines --
|
|
//
|
|
|
|
/* Performs a quick sort */
|
|
void
|
|
qsort (
|
|
void *base,
|
|
size_t num,
|
|
size_t width,
|
|
int ( *compare )(const void *, const void *)
|
|
)
|
|
{
|
|
VOID *Buffer;
|
|
|
|
ASSERT (base != NULL);
|
|
ASSERT (compare != NULL);
|
|
|
|
//
|
|
// Use CRT-style malloc to cover BS and RT memory allocation.
|
|
//
|
|
Buffer = CryptoPkg_BaseMemAllocation_malloc (width);
|
|
ASSERT (Buffer != NULL);
|
|
|
|
//
|
|
// Re-use PerformQuickSort() function Implementation in EDKII BaseSortLib.
|
|
//
|
|
QuickSortWorker (base, (UINTN)num, (UINTN)width, (SORT_COMPARE)compare, Buffer);
|
|
|
|
CryptoPkg_BaseMemAllocation_free (Buffer);
|
|
return;
|
|
}
|
|
|
|
//
|
|
// -- Process and Environment Control Routines --
|
|
//
|
|
|
|
/* Get a value from the current environment */
|
|
char *
|
|
getenv (
|
|
const char *varname
|
|
)
|
|
{
|
|
//
|
|
// Null getenv() function implementation to satisfy the linker, since there is
|
|
// no direct functionality logic dependency in present UEFI cases.
|
|
//
|
|
return NULL;
|
|
}
|
|
|
|
///* Get a value from the current environment */
|
|
//char *
|
|
//secure_getenv (
|
|
// const char *varname
|
|
// )
|
|
//{
|
|
// //
|
|
// // Null secure_getenv() function implementation to satisfy the linker, since
|
|
// // there is no direct functionality logic dependency in present UEFI cases.
|
|
// //
|
|
// // From the secure_getenv() manual: 'just like getenv() except that it
|
|
// // returns NULL in cases where "secure execution" is required'.
|
|
// //
|
|
// return NULL;
|
|
//}
|
|
//
|
|
////
|
|
//// -- Stream I/O Routines --
|
|
////
|
|
//
|
|
///* Write data to a stream */
|
|
//size_t
|
|
//fwrite (
|
|
// const void *buffer,
|
|
// size_t size,
|
|
// size_t count,
|
|
// FILE *stream
|
|
// )
|
|
//{
|
|
// return 0;
|
|
//}
|
|
//
|
|
//#ifdef __GNUC__
|
|
//
|
|
//typedef
|
|
//VOID
|
|
//(EFIAPI *NoReturnFuncPtr)(
|
|
// VOID
|
|
// ) __attribute__ ((__noreturn__));
|
|
//
|
|
//STATIC
|
|
//VOID
|
|
//EFIAPI
|
|
//NopFunction (
|
|
// VOID
|
|
// )
|
|
//{
|
|
//}
|
|
//
|
|
//void
|
|
//abort (
|
|
// void
|
|
// )
|
|
//{
|
|
// NoReturnFuncPtr NoReturnFunc;
|
|
//
|
|
// NoReturnFunc = (NoReturnFuncPtr)NopFunction;
|
|
//
|
|
// NoReturnFunc ();
|
|
//}
|
|
//
|
|
//#else
|
|
//
|
|
//void
|
|
//abort (
|
|
// void
|
|
// )
|
|
//{
|
|
// // Do nothing
|
|
//}
|
|
//
|
|
//#endif
|
|
//
|
|
//int
|
|
//fclose (
|
|
// FILE *f
|
|
// )
|
|
//{
|
|
// return 0;
|
|
//}
|
|
//
|
|
//FILE *
|
|
//fopen (
|
|
// const char *c,
|
|
// const char *m
|
|
// )
|
|
//{
|
|
// return NULL;
|
|
//}
|
|
//
|
|
//size_t
|
|
//fread (
|
|
// void *b,
|
|
// size_t c,
|
|
// size_t i,
|
|
// FILE *f
|
|
// )
|
|
//{
|
|
// return 0;
|
|
//}
|
|
//
|
|
//uid_t
|
|
//getuid (
|
|
// void
|
|
// )
|
|
//{
|
|
// return 0;
|
|
//}
|
|
//
|
|
//uid_t
|
|
//geteuid (
|
|
// void
|
|
// )
|
|
//{
|
|
// return 0;
|
|
//}
|
|
//
|
|
//gid_t
|
|
//getgid (
|
|
// void
|
|
// )
|
|
//{
|
|
// return 0;
|
|
//}
|
|
//
|
|
//gid_t
|
|
//getegid (
|
|
// void
|
|
// )
|
|
//{
|
|
// return 0;
|
|
//}
|
|
//
|
|
//int
|
|
//printf (
|
|
// char const *fmt,
|
|
// ...
|
|
// )
|
|
//{
|
|
// return 0;
|
|
//}
|