pnvram.c 7.46 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
/* Export pnvram and some variables for runtime */
/*
 *  GRUB  --  GRand Unified Bootloader
 *  Copyright (C) 2009  Free Software Foundation, Inc.
 *
 *  GRUB is free software: you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation, either version 3 of the License, or
 *  (at your option) any later version.
 *
 *  GRUB is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with GRUB.  If not, see <http://www.gnu.org/licenses/>.
 */

#include <grub/file.h>
#include <grub/err.h>
#include <grub/normal.h>
#include <grub/mm.h>
#include <grub/misc.h>
25
#include <grub/charset.h>
26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41
#include <grub/efiemu/efiemu.h>
#include <grub/efiemu/runtime.h>
#include <grub/extcmd.h>

/* Place for final location of variables */
static int nvram_handle = 0;
static int nvramsize_handle = 0;
static int high_monotonic_count_handle = 0;
static int timezone_handle = 0;
static int accuracy_handle = 0;
static int daylight_handle = 0;

static grub_size_t nvramsize;

/* Parse signed value */
static int
42
grub_strtosl (const char *arg, char **end, int base)
43 44 45 46 47 48
{
  if (arg[0] == '-')
    return -grub_strtoul (arg + 1, end, base);
  return grub_strtoul (arg, end, base);
}

49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80
static inline int
hextoval (char c)
{
  if (c >= '0' && c <= '9')
    return c - '0';
  if (c >= 'a' && c <= 'z')
    return c - 'a' + 10;
  if (c >= 'A' && c <= 'Z')
    return c - 'A' + 10;
  return 0;
}

static inline grub_err_t
unescape (char *in, char *out, char *outmax, int *len)
{
  char *ptr, *dptr;
  dptr = out;
  for (ptr = in; *ptr && dptr < outmax; )
    if (*ptr == '%' && ptr[1] && ptr[2])
      {
	*dptr = (hextoval (ptr[1]) << 4) | (hextoval (ptr[2]));
	ptr += 3;
	dptr++;
      }
    else
      {
	*dptr = *ptr;
	ptr++;
	dptr++;
      }
  if (dptr == outmax)
    return grub_error (GRUB_ERR_OUT_OF_MEMORY,
81 82
		       "too many NVRAM variables for reserved variable space."
		       " Try increasing EfiEmu.pnvram.size");
83 84 85 86
  *len = dptr - out;
  return 0;
}

87 88 89
/* Export stuff for efiemu */
static grub_err_t
nvram_set (void * data __attribute__ ((unused)))
90
{
91
  const char *env;
92
  /* Take definitive pointers */
93
  char *nvram = grub_efiemu_mm_obtain_request (nvram_handle);
94
  grub_uint32_t *nvramsize_def
95
    = grub_efiemu_mm_obtain_request (nvramsize_handle);
96
  grub_uint32_t *high_monotonic_count
97
    = grub_efiemu_mm_obtain_request (high_monotonic_count_handle);
98
  grub_int16_t *timezone
99
    = grub_efiemu_mm_obtain_request (timezone_handle);
100
  grub_uint8_t *daylight
101
    = grub_efiemu_mm_obtain_request (daylight_handle);
102
  grub_uint32_t *accuracy
103
    = grub_efiemu_mm_obtain_request (accuracy_handle);
104
  char *nvramptr;
105
  struct grub_env_var *var;
106

107 108 109 110 111 112 113 114 115 116 117 118 119 120 121
  /* Copy to definitive loaction */
  grub_dprintf ("efiemu", "preparing pnvram\n");

  env = grub_env_get ("EfiEmu.pnvram.high_monotonic_count");
  *high_monotonic_count = env ? grub_strtoul (env, 0, 0) : 1;
  env = grub_env_get ("EfiEmu.pnvram.timezone");
  *timezone = env ? grub_strtosl (env, 0, 0) : GRUB_EFI_UNSPECIFIED_TIMEZONE;
  env = grub_env_get ("EfiEmu.pnvram.accuracy");
  *accuracy = env ? grub_strtoul (env, 0, 0) : 50000000;
  env = grub_env_get ("EfiEmu.pnvram.daylight");
  *daylight = env ? grub_strtoul (env, 0, 0) : 0;

  nvramptr = nvram;
  grub_memset (nvram, 0, nvramsize);
  FOR_SORTED_ENV (var)
122 123 124 125
  {
    char *guid, *attr, *name, *varname;
    struct efi_variable *efivar;
    int len = 0;
126 127
    int i;
    grub_uint64_t guidcomp;
128 129 130

    if (grub_memcmp (var->name, "EfiEmu.pnvram.",
		     sizeof ("EfiEmu.pnvram.") - 1) != 0)
131
      continue;
132 133 134 135 136

    guid = var->name + sizeof ("EfiEmu.pnvram.") - 1;

    attr = grub_strchr (guid, '.');
    if (!attr)
137
      continue;
138 139 140 141
    attr++;

    name = grub_strchr (attr, '.');
    if (!name)
142
      continue;
143 144 145 146
    name++;

    efivar = (struct efi_variable *) nvramptr;
    if (nvramptr - nvram + sizeof (struct efi_variable) > nvramsize)
147 148 149
      return grub_error (GRUB_ERR_OUT_OF_MEMORY,
			 "too many NVRAM variables for reserved variable space."
			 " Try increasing EfiEmu.pnvram.size");
150 151 152 153 154

    nvramptr += sizeof (struct efi_variable);

    efivar->guid.data1 = grub_cpu_to_le32 (grub_strtoul (guid, &guid, 16));
    if (*guid != '-')
155
      continue;
156 157 158 159
    guid++;

    efivar->guid.data2 = grub_cpu_to_le16 (grub_strtoul (guid, &guid, 16));
    if (*guid != '-')
160
      continue;
161 162 163 164
    guid++;

    efivar->guid.data3 = grub_cpu_to_le16 (grub_strtoul (guid, &guid, 16));
    if (*guid != '-')
165
      continue;
166 167
    guid++;

168 169 170
    guidcomp = grub_strtoull (guid, 0, 16);
    for (i = 0; i < 8; i++)
      efivar->guid.data4[i] = (guidcomp >> (56 - 8 * i)) & 0xff;
171 172 173 174 175

    efivar->attributes = grub_strtoull (attr, 0, 16);

    varname = grub_malloc (grub_strlen (name) + 1);
    if (! varname)
176
      return grub_errno;
177 178

    if (unescape (name, varname, varname + grub_strlen (name) + 1, &len))
179
      break;
180 181 182 183 184 185 186 187 188 189 190 191 192

    len = grub_utf8_to_utf16 ((grub_uint16_t *) nvramptr,
			      (nvramsize - (nvramptr - nvram)) / 2,
			      (grub_uint8_t *) varname, len, NULL);

    nvramptr += 2 * len;
    *((grub_uint16_t *) nvramptr) = 0;
    nvramptr += 2;
    efivar->namelen = 2 * len + 2;

    if (unescape (var->value, nvramptr, nvram + nvramsize, &len))
      {
	efivar->namelen = 0;
193
	break;
194 195 196 197 198 199 200 201
      }

    nvramptr += len;

    efivar->size = len;
  }
  if (grub_errno)
    return grub_errno;
202

203 204 205 206 207
  *nvramsize_def = nvramsize;

  /* Register symbols */
  grub_efiemu_register_symbol ("efiemu_variables", nvram_handle, 0);
  grub_efiemu_register_symbol ("efiemu_varsize", nvramsize_handle, 0);
208
  grub_efiemu_register_symbol ("efiemu_high_monotonic_count",
209 210 211
			       high_monotonic_count_handle, 0);
  grub_efiemu_register_symbol ("efiemu_time_zone", timezone_handle, 0);
  grub_efiemu_register_symbol ("efiemu_time_daylight", daylight_handle, 0);
212
  grub_efiemu_register_symbol ("efiemu_time_accuracy",
213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228
			       accuracy_handle, 0);

  return GRUB_ERR_NONE;
}

static void
nvram_unload (void * data __attribute__ ((unused)))
{
  grub_efiemu_mm_return_request (nvram_handle);
  grub_efiemu_mm_return_request (nvramsize_handle);
  grub_efiemu_mm_return_request (high_monotonic_count_handle);
  grub_efiemu_mm_return_request (timezone_handle);
  grub_efiemu_mm_return_request (accuracy_handle);
  grub_efiemu_mm_return_request (daylight_handle);
}

229 230
grub_err_t
grub_efiemu_pnvram (void)
231
{
232 233
  const char *size;
  grub_err_t err;
234

235
  nvramsize = 0;
236

237 238 239
  size = grub_env_get ("EfiEmu.pnvram.size");
  if (size)
    nvramsize = grub_strtoul (size, 0, 0);
240

241 242
  if (!nvramsize)
    nvramsize = 2048;
243 244 245

  err = grub_efiemu_register_prepare_hook (nvram_set, nvram_unload, 0);
  if (err)
246 247
    return err;

248 249
  nvram_handle
    = grub_efiemu_request_memalign (1, nvramsize,
250
				    GRUB_EFI_RUNTIME_SERVICES_DATA);
251 252
  nvramsize_handle
    = grub_efiemu_request_memalign (1, sizeof (grub_uint32_t),
253 254
				    GRUB_EFI_RUNTIME_SERVICES_DATA);
  high_monotonic_count_handle
255
    = grub_efiemu_request_memalign (1, sizeof (grub_uint32_t),
256 257
				    GRUB_EFI_RUNTIME_SERVICES_DATA);
  timezone_handle
258
    = grub_efiemu_request_memalign (1, sizeof (grub_uint16_t),
259 260
				    GRUB_EFI_RUNTIME_SERVICES_DATA);
  daylight_handle
261
    = grub_efiemu_request_memalign (1, sizeof (grub_uint8_t),
262 263
				    GRUB_EFI_RUNTIME_SERVICES_DATA);
  accuracy_handle
264
    = grub_efiemu_request_memalign (1, sizeof (grub_uint32_t),
265 266 267 268
				    GRUB_EFI_RUNTIME_SERVICES_DATA);

  return GRUB_ERR_NONE;
}