We need to avoid clobbering existing table between starting of chunk movers and the moment we install target page table. Generate temporary table for this rather than hoping that we don't clobber existing one. Fixes 64-bit GhostBSD on 64-bit EFI. Signed-off-by: Vladimir Serbinenko <phcoder@gmail.com> Reviewed-by: Daniel Kiper <daniel.kiper@oracle.com>
154 lines
4.3 KiB
C
154 lines
4.3 KiB
C
/*
|
|
* 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/mm.h>
|
|
#include <grub/misc.h>
|
|
|
|
#include <grub/types.h>
|
|
#include <grub/types.h>
|
|
#include <grub/err.h>
|
|
#include <grub/cache.h>
|
|
|
|
#include <grub/mips/relocator.h>
|
|
#include <grub/relocator_private.h>
|
|
|
|
/* Do we need mips64? */
|
|
|
|
extern grub_uint8_t grub_relocator_forward_start;
|
|
extern grub_uint8_t grub_relocator_forward_end;
|
|
extern grub_uint8_t grub_relocator_backward_start;
|
|
extern grub_uint8_t grub_relocator_backward_end;
|
|
|
|
#define REGW_SIZEOF (2 * sizeof (grub_uint32_t))
|
|
#define JUMP_SIZEOF (2 * sizeof (grub_uint32_t))
|
|
|
|
#define RELOCATOR_SRC_SIZEOF(x) (&grub_relocator_##x##_end \
|
|
- &grub_relocator_##x##_start)
|
|
#define RELOCATOR_SIZEOF(x) (RELOCATOR_SRC_SIZEOF(x) \
|
|
+ REGW_SIZEOF * 3)
|
|
grub_size_t grub_relocator_align = sizeof (grub_uint32_t);
|
|
grub_size_t grub_relocator_forward_size;
|
|
grub_size_t grub_relocator_backward_size;
|
|
grub_size_t grub_relocator_jumper_size = JUMP_SIZEOF + REGW_SIZEOF;
|
|
grub_size_t grub_relocator_preamble_size = 0;
|
|
|
|
void
|
|
grub_cpu_relocator_init (void)
|
|
{
|
|
grub_relocator_forward_size = RELOCATOR_SIZEOF(forward);
|
|
grub_relocator_backward_size = RELOCATOR_SIZEOF(backward);
|
|
}
|
|
|
|
void
|
|
grub_cpu_relocator_preamble (void *rels __attribute__ ((unused)))
|
|
{
|
|
}
|
|
|
|
static void
|
|
write_reg (int regn, grub_uint32_t val, void **target)
|
|
{
|
|
/* lui $r, (val+0x8000). */
|
|
*(grub_uint32_t *) *target = ((0x3c00 | regn) << 16) | ((val + 0x8000) >> 16);
|
|
*target = ((grub_uint32_t *) *target) + 1;
|
|
/* addiu $r, $r, val. */
|
|
*(grub_uint32_t *) *target = (((0x2400 | regn << 5 | regn) << 16)
|
|
| (val & 0xffff));
|
|
*target = ((grub_uint32_t *) *target) + 1;
|
|
}
|
|
|
|
static void
|
|
write_jump (int regn, void **target)
|
|
{
|
|
/* j $r. */
|
|
*(grub_uint32_t *) *target = (regn<<21) | 0x8;
|
|
*target = ((grub_uint32_t *) *target) + 1;
|
|
/* nop. */
|
|
*(grub_uint32_t *) *target = 0;
|
|
*target = ((grub_uint32_t *) *target) + 1;
|
|
}
|
|
|
|
void
|
|
grub_cpu_relocator_jumper (void *rels, grub_addr_t addr)
|
|
{
|
|
write_reg (1, addr, &rels);
|
|
write_jump (1, &rels);
|
|
}
|
|
|
|
void
|
|
grub_cpu_relocator_backward (void *ptr0, void *src, void *dest,
|
|
grub_size_t size)
|
|
{
|
|
void *ptr = ptr0;
|
|
write_reg (8, (grub_uint32_t) src, &ptr);
|
|
write_reg (9, (grub_uint32_t) dest, &ptr);
|
|
write_reg (10, (grub_uint32_t) size, &ptr);
|
|
grub_memcpy (ptr, &grub_relocator_backward_start,
|
|
RELOCATOR_SRC_SIZEOF (backward));
|
|
}
|
|
|
|
void
|
|
grub_cpu_relocator_forward (void *ptr0, void *src, void *dest,
|
|
grub_size_t size)
|
|
{
|
|
void *ptr = ptr0;
|
|
write_reg (8, (grub_uint32_t) src, &ptr);
|
|
write_reg (9, (grub_uint32_t) dest, &ptr);
|
|
write_reg (10, (grub_uint32_t) size, &ptr);
|
|
grub_memcpy (ptr, &grub_relocator_forward_start,
|
|
RELOCATOR_SRC_SIZEOF (forward));
|
|
}
|
|
|
|
grub_err_t
|
|
grub_relocator32_boot (struct grub_relocator *rel,
|
|
struct grub_relocator32_state state)
|
|
{
|
|
grub_relocator_chunk_t ch;
|
|
void *ptr;
|
|
grub_err_t err;
|
|
void *relst;
|
|
grub_size_t relsize;
|
|
grub_size_t stateset_size = 31 * REGW_SIZEOF + JUMP_SIZEOF;
|
|
unsigned i;
|
|
grub_addr_t vtarget;
|
|
|
|
err = grub_relocator_alloc_chunk_align (rel, &ch, 0, UP_TO_TOP32 (stateset_size),
|
|
stateset_size, sizeof (grub_uint32_t),
|
|
GRUB_RELOCATOR_PREFERENCE_NONE, 0);
|
|
if (err)
|
|
return err;
|
|
|
|
ptr = get_virtual_current_address (ch);
|
|
for (i = 1; i < 32; i++)
|
|
write_reg (i, state.gpr[i], &ptr);
|
|
write_jump (state.jumpreg, &ptr);
|
|
|
|
vtarget = (grub_addr_t) grub_map_memory (get_physical_target_address (ch),
|
|
stateset_size);
|
|
|
|
err = grub_relocator_prepare_relocs (rel, vtarget, &relst, &relsize);
|
|
if (err)
|
|
return err;
|
|
|
|
grub_arch_sync_caches ((void *) relst, relsize);
|
|
|
|
((void (*) (void)) relst) ();
|
|
|
|
/* Not reached. */
|
|
return GRUB_ERR_NONE;
|
|
}
|