mirror of
				https://git.proxmox.com/git/mirror_edk2
				synced 2025-11-04 09:12:31 +00:00 
			
		
		
		
	Include/ctype.h:             Function declaration and Macro definition of isDirSep
Include/unistd.h:            Declarations added from EfiSysCall.h
Include/utime.h:             New file.  For the Posix utime() function.
Include/sys/_ctype.h:        Update character class bit maps.
Include/sys/EfiSysCall.h:    Move declarations to unistd.h
Include/sys/fcntl.h:         Improve comments.  Add UEFI-specific macros.
Include/sys/filio.h:         Remove declarations for unsupported file ioctls.
Include/sys/stat.h:          Fix flags.  Add macros and declarations.
Include/sys/time.h:          Add declarations for new functions Tm2Efi() and Time2Efi().
Include/sys/types.h:         Use EFI-specific instead of BSD-specific definitions for typedefs.
Include/sys/unistd.h:        Delete inappropriate content.  Guard macro definitions.
LibC/Locale/setlocale.c
LibC/Stdio/{fdopen.c, findfp.c, fopen.c, freopen.c, gettemp.c, makebuf.c, mktemp.c,
            remove.c, stdio.c, tempnam.c, tmpfile.c, tmpnam.c}
LibC/Time/{itimer.c, ZoneProc.c}
LibC/Uefi/SysCalls.c
LibC/Uefi/Devices/Console/daConsole.c
LibC/Uefi/Devices/UefiShell/daShell.c
PosixLib/Gen/readdir.c
                             Include unistd.h instead of EfiSysCall.h
LibC/Ctype/CClass.c:         Character classification function implementation for isDirSep.
LibC/Ctype/iCtype.c:         Update character classification and case conversion tables.
LibC/Time/TimeEfi.c:         Improve comments.  Implement new functions Tm2Efi() and Time2Efi().
LibC/Uefi/StubFunctions.c:   Add missing include.  Cosmetic changes to declarations.
LibC/Uefi/SysCalls.c:        Add support function for utime().
LibC/Uefi/Uefi.inf:          Add LibGen library class dependency.
LibC/Uefi/Xform.c:           Enhance Omode2EFI().
LibC/Uefi/Devices/UefiShell/daShell.c:    Enhance da_ShellMkdir.  Implement da_ShellIoctl to set file times.
PosixLib/Gen/access.c:       New file.  Implement the access() function.
PosixLib/Gen/dirname.c:      Enhance to use isDirSep and differentiate between the device, path, and filename components of UEFI Shell-style paths.
PosixLib/Gen/utime.c:        New file.  Implement the utime() function.
PosixLib/Gen/LibGen.inf:     Change MODULE_TYPE.  Add new files.
Signed-off-by: darylm503
Reviewed-by: geekboy15a
Reviewed-by: jljusten
Reviewed-by: Rahul Khana
Reviewed-by: leegrosenbaum
git-svn-id: https://edk2.svn.sourceforge.net/svnroot/edk2/trunk/edk2@12800 6f19259b-4bc3-4df7-8a09-765794883524
		
	
			
		
			
				
	
	
		
			425 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			425 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/** @file
 | 
						|
  Worker functions for the setlocale function.
 | 
						|
 | 
						|
  Copyright (c) 2010 - 2011, Intel Corporation. All rights reserved.<BR>
 | 
						|
  This program and the accompanying materials are licensed and made available under
 | 
						|
  the terms and conditions of the BSD License that accompanies this distribution.
 | 
						|
  The full text of the license may be found at
 | 
						|
  http://opensource.org/licenses/bsd-license.
 | 
						|
 | 
						|
  THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
 | 
						|
  WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
 | 
						|
 | 
						|
 * Copyright (c) 1991, 1993
 | 
						|
 *  The Regents of the University of California.  All rights reserved.
 | 
						|
 *
 | 
						|
 * This code is derived from software contributed to Berkeley by
 | 
						|
 * Paul Borman at Krystal Technologies.
 | 
						|
 *
 | 
						|
 * Redistribution and use in source and binary forms, with or without
 | 
						|
 * modification, are permitted provided that the following conditions
 | 
						|
 * are met:
 | 
						|
 * 1. Redistributions of source code must retain the above copyright
 | 
						|
 *    notice, this list of conditions and the following disclaimer.
 | 
						|
 * 2. Redistributions in binary form must reproduce the above copyright
 | 
						|
 *    notice, this list of conditions and the following disclaimer in the
 | 
						|
 *    documentation and/or other materials provided with the distribution.
 | 
						|
 * 3. Neither the name of the University nor the names of its contributors
 | 
						|
 *    may be used to endorse or promote products derived from this software
 | 
						|
 *    without specific prior written permission.
 | 
						|
 *
 | 
						|
 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
 | 
						|
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 | 
						|
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 | 
						|
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
 | 
						|
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 | 
						|
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 | 
						|
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 | 
						|
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 | 
						|
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 | 
						|
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 | 
						|
 * SUCH DAMAGE.
 | 
						|
 | 
						|
    setlocale.c 8.1 (Berkeley) 7/4/93
 | 
						|
 *  NetBSD: setlocale.c,v 1.50 2006/02/16 19:19:49 tnozaki Exp
 | 
						|
**/
 | 
						|
#include  <LibConfig.h>
 | 
						|
#include  <sys/EfiCdefs.h>
 | 
						|
 | 
						|
#if defined(_MSC_VER)
 | 
						|
  // Disable warnings about assignment within conditional expressions.
 | 
						|
  #pragma warning ( disable : 4706 )
 | 
						|
#endif
 | 
						|
 | 
						|
#define _CTYPE_PRIVATE
 | 
						|
 | 
						|
#include  "namespace.h"
 | 
						|
#include  <sys/localedef.h>
 | 
						|
#include  <sys/types.h>
 | 
						|
#include  <sys/stat.h>
 | 
						|
#include  <assert.h>
 | 
						|
#include  <limits.h>
 | 
						|
#include  <ctype.h>
 | 
						|
#define __SETLOCALE_SOURCE__
 | 
						|
#include  <locale.h>
 | 
						|
#include  <paths.h>
 | 
						|
#include  <stdio.h>
 | 
						|
#include  <stdlib.h>
 | 
						|
#include  <string.h>
 | 
						|
#include  <unistd.h>
 | 
						|
#include  "rune.h"
 | 
						|
#ifdef WITH_RUNE
 | 
						|
  #include "rune_local.h"
 | 
						|
#else
 | 
						|
  #include "ctypeio.h"
 | 
						|
#endif
 | 
						|
 | 
						|
#ifdef CITRUS
 | 
						|
  #include <citrus/citrus_namespace.h>
 | 
						|
  #include <citrus/citrus_region.h>
 | 
						|
  #include <citrus/citrus_lookup.h>
 | 
						|
  #include <citrus/citrus_bcs.h>
 | 
						|
#else
 | 
						|
  #include "aliasname_local.h"
 | 
						|
  #define _lookup_alias(p, a, b, s, c)  __unaliasname((p), (a), (b), (s))
 | 
						|
  #define _bcs_strcasecmp(a, b)   strcasecmp((a), (b))
 | 
						|
#endif
 | 
						|
 | 
						|
#define _LOCALE_SYM_FORCE "/force"
 | 
						|
 | 
						|
#ifndef WITH_RUNE
 | 
						|
  const char *_PathLocale = NULL;
 | 
						|
#endif
 | 
						|
 | 
						|
/** Category names for getenv(). **/
 | 
						|
static const char *const categories[_LC_LAST] = {
 | 
						|
    "LC_ALL",
 | 
						|
    "LC_COLLATE",
 | 
						|
    "LC_CTYPE",
 | 
						|
    "LC_MONETARY",
 | 
						|
    "LC_NUMERIC",
 | 
						|
    "LC_TIME",
 | 
						|
    "LC_MESSAGES"
 | 
						|
};
 | 
						|
 | 
						|
/** Current locales for each category.  **/
 | 
						|
static char current_categories[_LC_LAST][32] = {
 | 
						|
    "C",
 | 
						|
    "C",
 | 
						|
    "C",
 | 
						|
    "C",
 | 
						|
    "C",
 | 
						|
    "C",
 | 
						|
    "C"
 | 
						|
};
 | 
						|
 | 
						|
/*
 | 
						|
 * The locales we are going to try and load
 | 
						|
 */
 | 
						|
static char new_categories[_LC_LAST][32];
 | 
						|
 | 
						|
static char current_locale_string[_LC_LAST * 33];
 | 
						|
 | 
						|
static char *currentlocale(void);
 | 
						|
static void revert_to_default(int);
 | 
						|
static int force_locale_enable(int);
 | 
						|
static int load_locale_sub(int, const char *, int);
 | 
						|
static char *loadlocale(int);
 | 
						|
static const char *__get_locale_env(int);
 | 
						|
 | 
						|
char *
 | 
						|
__setlocale(int category, const char *locale)
 | 
						|
{
 | 
						|
  int i, loadlocale_success;
 | 
						|
  size_t len;
 | 
						|
  const char *env, *r;
 | 
						|
 | 
						|
  //if (issetugid() ||
 | 
						|
  //    (!_PathLocale && !(_PathLocale = getenv("PATH_LOCALE"))))
 | 
						|
  //  _PathLocale = _PATH_LOCALE;
 | 
						|
 | 
						|
  if (category < 0 || category >= _LC_LAST)
 | 
						|
    return (NULL);
 | 
						|
 | 
						|
  if (!locale)
 | 
						|
    return (category ?
 | 
						|
        current_categories[category] : currentlocale());
 | 
						|
 | 
						|
  /*
 | 
						|
   * Default to the current locale for everything.
 | 
						|
   */
 | 
						|
  for (i = 1; i < _LC_LAST; ++i)
 | 
						|
    (void)strncpyX(new_categories[i], current_categories[i],
 | 
						|
        sizeof(new_categories[i]));
 | 
						|
 | 
						|
  /*
 | 
						|
   * Now go fill up new_categories from the locale argument
 | 
						|
   */
 | 
						|
  if (!*locale) {
 | 
						|
    if (category == LC_ALL) {
 | 
						|
      for (i = 1; i < _LC_LAST; ++i) {
 | 
						|
        env = __get_locale_env(i);
 | 
						|
        (void)strncpyX(new_categories[i], env,
 | 
						|
            sizeof(new_categories[i]));
 | 
						|
      }
 | 
						|
    }
 | 
						|
    else {
 | 
						|
      env = __get_locale_env(category);
 | 
						|
      (void)strncpyX(new_categories[category], env,
 | 
						|
        sizeof(new_categories[category]));
 | 
						|
    }
 | 
						|
  } else if (category) {
 | 
						|
    (void)strncpyX(new_categories[category], locale,
 | 
						|
        sizeof(new_categories[category]));
 | 
						|
  } else {
 | 
						|
    if ((r = strchr(locale, '/')) == 0) {
 | 
						|
      for (i = 1; i < _LC_LAST; ++i) {
 | 
						|
        (void)strncpyX(new_categories[i], locale,
 | 
						|
            sizeof(new_categories[i]));
 | 
						|
      }
 | 
						|
    } else {
 | 
						|
      for (i = 1;;) {
 | 
						|
        _DIAGASSERT(*r == '/' || *r == 0);
 | 
						|
        _DIAGASSERT(*locale != 0);
 | 
						|
        if (*locale == '/')
 | 
						|
          return (NULL);  /* invalid format. */
 | 
						|
        len = r - locale;
 | 
						|
        if (len + 1 > sizeof(new_categories[i]))
 | 
						|
          return (NULL);  /* too long */
 | 
						|
        (void)memcpy(new_categories[i], locale, len);
 | 
						|
        new_categories[i][len] = '\0';
 | 
						|
        if (*r == 0)
 | 
						|
          break;
 | 
						|
        _DIAGASSERT(*r == '/');
 | 
						|
        if (*(locale = ++r) == 0)
 | 
						|
          /* slash followed by NUL */
 | 
						|
          return (NULL);
 | 
						|
        /* skip until NUL or '/' */
 | 
						|
        while (*r && *r != '/')
 | 
						|
          r++;
 | 
						|
        if (++i == _LC_LAST)
 | 
						|
          return (NULL);  /* too many slashes. */
 | 
						|
      }
 | 
						|
      if (i + 1 != _LC_LAST)
 | 
						|
        return (NULL);  /* too few slashes. */
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (category)
 | 
						|
    return (loadlocale(category));
 | 
						|
 | 
						|
  loadlocale_success = 0;
 | 
						|
  for (i = 1; i < _LC_LAST; ++i) {
 | 
						|
    if (loadlocale(i) != NULL)
 | 
						|
      loadlocale_success = 1;
 | 
						|
  }
 | 
						|
 | 
						|
  /*
 | 
						|
   * If all categories failed, return NULL; we don't need to back
 | 
						|
   * changes off, since none happened.
 | 
						|
   */
 | 
						|
  if (!loadlocale_success)
 | 
						|
    return NULL;
 | 
						|
 | 
						|
  return (currentlocale());
 | 
						|
}
 | 
						|
 | 
						|
static char *
 | 
						|
currentlocale()
 | 
						|
{
 | 
						|
  int i;
 | 
						|
 | 
						|
  (void)strncpyX(current_locale_string, current_categories[1],
 | 
						|
      sizeof(current_locale_string));
 | 
						|
 | 
						|
  for (i = 2; i < _LC_LAST; ++i)
 | 
						|
    if (strcmp(current_categories[1], current_categories[i])) {
 | 
						|
      (void)snprintf(current_locale_string,
 | 
						|
          sizeof(current_locale_string), "%s/%s/%s/%s/%s/%s",
 | 
						|
          current_categories[1], current_categories[2],
 | 
						|
          current_categories[3], current_categories[4],
 | 
						|
          current_categories[5], current_categories[6]);
 | 
						|
      break;
 | 
						|
    }
 | 
						|
  return (current_locale_string);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
revert_to_default(int category)
 | 
						|
{
 | 
						|
  switch (category) {
 | 
						|
  case LC_CTYPE:
 | 
						|
#ifdef WITH_RUNE
 | 
						|
    (void)_xpg4_setrunelocale("C");
 | 
						|
    (void)__runetable_to_netbsd_ctype("C");
 | 
						|
#else
 | 
						|
    if (_cClass != _C_CharClassTable) {
 | 
						|
      /* LINTED const castaway */
 | 
						|
      free((void *)_cClass);
 | 
						|
      _cClass = _C_CharClassTable;
 | 
						|
    }
 | 
						|
    if (_uConvT != _C_ToUpperTable) {
 | 
						|
      /* LINTED const castaway */
 | 
						|
      free((void *)_uConvT);
 | 
						|
      _uConvT = _C_ToUpperTable;
 | 
						|
    }
 | 
						|
    if (_lConvT != _C_ToLowerTable) {
 | 
						|
      /* LINTED const castaway */
 | 
						|
      free((void *)_lConvT);
 | 
						|
      _lConvT = _C_ToLowerTable;
 | 
						|
    }
 | 
						|
#endif
 | 
						|
    break;
 | 
						|
  case LC_MESSAGES:
 | 
						|
  case LC_COLLATE:
 | 
						|
  case LC_MONETARY:
 | 
						|
  case LC_NUMERIC:
 | 
						|
  case LC_TIME:
 | 
						|
    break;
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
force_locale_enable(int category)
 | 
						|
{
 | 
						|
  revert_to_default(category);
 | 
						|
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
load_locale_sub(
 | 
						|
  int category,
 | 
						|
  const char *locname,
 | 
						|
  int isspecial
 | 
						|
  )
 | 
						|
{
 | 
						|
  char name[PATH_MAX];
 | 
						|
 | 
						|
  /* check for the default locales */
 | 
						|
  if (!strcmp(new_categories[category], "C") ||
 | 
						|
      !strcmp(new_categories[category], "POSIX")) {
 | 
						|
    revert_to_default(category);
 | 
						|
    return 0;
 | 
						|
  }
 | 
						|
 | 
						|
  /* check whether special symbol */
 | 
						|
  if (isspecial && _bcs_strcasecmp(locname, _LOCALE_SYM_FORCE) == 0)
 | 
						|
    return force_locale_enable(category);
 | 
						|
 | 
						|
  /* sanity check */
 | 
						|
  if (strchr(locname, '/') != NULL)
 | 
						|
    return -1;
 | 
						|
 | 
						|
  (void)snprintf(name, sizeof(name), "%s/%s/%s",
 | 
						|
           _PathLocale, locname, categories[category]);
 | 
						|
 | 
						|
  switch (category) {
 | 
						|
  case LC_CTYPE:
 | 
						|
#ifdef WITH_RUNE
 | 
						|
    if (_xpg4_setrunelocale(__UNCONST(locname)))
 | 
						|
      return -1;
 | 
						|
    if (__runetable_to_netbsd_ctype(locname)) {
 | 
						|
      /* very unfortunate, but need to go to "C" locale */
 | 
						|
      revert_to_default(category);
 | 
						|
      return -1;
 | 
						|
    }
 | 
						|
#else
 | 
						|
    if (!__loadctype(name))
 | 
						|
      return -1;
 | 
						|
#endif
 | 
						|
    break;
 | 
						|
 | 
						|
  case LC_MESSAGES:
 | 
						|
    /*
 | 
						|
     * XXX we don't have LC_MESSAGES support yet,
 | 
						|
     * but catopen may use the value of LC_MESSAGES category.
 | 
						|
     * so return successfully if locale directory is present.
 | 
						|
     */
 | 
						|
    (void)snprintf(name, sizeof(name), "%s/%s",
 | 
						|
      _PathLocale, locname);
 | 
						|
    /* local */
 | 
						|
    {
 | 
						|
      struct stat st;
 | 
						|
      if (stat(name, &st) < 0)
 | 
						|
        return -1;
 | 
						|
      if (!S_ISDIR(st.st_mode))
 | 
						|
        return -1;
 | 
						|
    }
 | 
						|
    break;
 | 
						|
 | 
						|
  case LC_COLLATE:
 | 
						|
  case LC_MONETARY:
 | 
						|
  case LC_NUMERIC:
 | 
						|
  case LC_TIME:
 | 
						|
    return -1;
 | 
						|
  }
 | 
						|
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
static char *
 | 
						|
loadlocale(int category)
 | 
						|
{
 | 
						|
  //char aliaspath[PATH_MAX], loccat[PATH_MAX], buf[PATH_MAX];
 | 
						|
  //const char *alias;
 | 
						|
 | 
						|
  _DIAGASSERT(0 < category && category < _LC_LAST);
 | 
						|
 | 
						|
  if (strcmp(new_categories[category], current_categories[category]) == 0)
 | 
						|
    return (current_categories[category]);
 | 
						|
 | 
						|
  /* (1) non-aliased file */
 | 
						|
  if (!load_locale_sub(category, new_categories[category], 0))
 | 
						|
    goto success;
 | 
						|
 | 
						|
  ///* (2) lookup locname/catname type alias */
 | 
						|
  //(void)snprintf(aliaspath, sizeof(aliaspath),
 | 
						|
  //         "%s/" _LOCALE_ALIAS_NAME, _PathLocale);
 | 
						|
  //(void)snprintf(loccat, sizeof(loccat), "%s/%s",
 | 
						|
  //         new_categories[category], categories[category]);
 | 
						|
  //alias = _lookup_alias(aliaspath, loccat, buf, sizeof(buf),
 | 
						|
  //          _LOOKUP_CASE_SENSITIVE);
 | 
						|
  //if (!load_locale_sub(category, alias, 1))
 | 
						|
  //  goto success;
 | 
						|
 | 
						|
  ///* (3) lookup locname type alias */
 | 
						|
  //alias = _lookup_alias(aliaspath, new_categories[category],
 | 
						|
  //          buf, sizeof(buf), _LOOKUP_CASE_SENSITIVE);
 | 
						|
  //if (!load_locale_sub(category, alias, 1))
 | 
						|
  //  goto success;
 | 
						|
 | 
						|
  return NULL;
 | 
						|
 | 
						|
success:
 | 
						|
  (void)strncpyX(current_categories[category],
 | 
						|
    new_categories[category],
 | 
						|
    sizeof(current_categories[category]));
 | 
						|
  return current_categories[category];
 | 
						|
}
 | 
						|
 | 
						|
static const char *
 | 
						|
__get_locale_env(int category)
 | 
						|
{
 | 
						|
  const char *env;
 | 
						|
 | 
						|
  //_DIAGASSERT(category != LC_ALL);
 | 
						|
 | 
						|
  ///* 1. check LC_ALL. */
 | 
						|
  //env = getenv(categories[0]);
 | 
						|
 | 
						|
  ///* 2. check LC_* */
 | 
						|
  //if (!env || !*env)
 | 
						|
  //  env = getenv(categories[category]);
 | 
						|
 | 
						|
  ///* 3. check LANG */
 | 
						|
  //if (!env || !*env)
 | 
						|
  //  env = getenv("LANG");
 | 
						|
 | 
						|
  ///* 4. if none is set, fall to "C" */
 | 
						|
  //if (!env || !*env || strchr(env, '/'))
 | 
						|
    env = "C";
 | 
						|
 | 
						|
  return env;
 | 
						|
}
 |