~hamo/ubuntu/precise/grub2/grub2.hi_res

« back to all changes in this revision

Viewing changes to grub-core/lib/mips/relocator.c

  • Committer: Bazaar Package Importer
  • Author(s): Colin Watson, Colin Watson, Robert Millan, Updated translations
  • Date: 2010-11-22 12:24:56 UTC
  • mfrom: (1.26.4 upstream) (17.3.36 sid)
  • mto: (17.3.43 sid)
  • mto: This revision was merged to the branch mainline in revision 89.
  • Revision ID: james.westby@ubuntu.com-20101122122456-y82z3sfb7k4zfdcc
Tags: 1.99~20101122-1
[ Colin Watson ]
* New Bazaar snapshot.  Too many changes to list in full, but some of the
  more user-visible ones are as follows:
  - GRUB script:
    + Function parameters, "break", "continue", "shift", "setparams",
      "return", and "!".
    + "export" command supports multiple variable names.
    + Multi-line quoted strings support.
    + Wildcard expansion.
  - sendkey support.
  - USB hotunplugging and USB serial support.
  - Rename CD-ROM to cd on BIOS.
  - Add new --boot-directory option to grub-install, grub-reboot, and
    grub-set-default; the old --root-directory option is still accepted
    but was often confusing.
  - Basic btrfs detection/UUID support (but no file reading yet).
  - bash-completion for utilities.
  - If a device is listed in device.map, always assume that it is
    BIOS-visible rather than using extra layers such as LVM or RAID.
  - Add grub-mknetdir script (closes: #550658).
  - Remove deprecated "root" command.
  - Handle RAID devices containing virtio components.
  - GRUB Legacy configuration file support (via grub-menulst2cfg).
  - Keyboard layout support (via grub-mklayout and grub-kbdcomp).
  - Check generated grub.cfg for syntax errors before saving.
  - Pause execution for at most ten seconds if any errors are displayed,
    so that the user has a chance to see them.
  - Support submenus.
  - Write embedding zone using Reed-Solomon, so that it's robust against
    being partially overwritten (closes: #550702, #591416, #593347).
  - GRUB_DISABLE_LINUX_RECOVERY and GRUB_DISABLE_NETBSD_RECOVERY merged
    into a single GRUB_DISABLE_RECOVERY variable.
  - Fix loader memory allocation failure (closes: #551627).
  - Don't call savedefault on recovery entries (closes: #589325).
  - Support triple-indirect blocks on ext2 (closes: #543924).
  - Recognise DDF1 fake RAID (closes: #603354).

[ Robert Millan ]
* Use dpkg architecture wildcards.

[ Updated translations ]
* Slovenian (Vanja Cvelbar).  Closes: #604003
* Dzongkha (dawa pemo via Tenzin Dendup).  Closes: #604102

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
/*
 
2
 *  GRUB  --  GRand Unified Bootloader
 
3
 *  Copyright (C) 2009  Free Software Foundation, Inc.
 
4
 *
 
5
 *  GRUB is free software: you can redistribute it and/or modify
 
6
 *  it under the terms of the GNU General Public License as published by
 
7
 *  the Free Software Foundation, either version 3 of the License, or
 
8
 *  (at your option) any later version.
 
9
 *
 
10
 *  GRUB is distributed in the hope that it will be useful,
 
11
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 
12
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
13
 *  GNU General Public License for more details.
 
14
 *
 
15
 *  You should have received a copy of the GNU General Public License
 
16
 *  along with GRUB.  If not, see <http://www.gnu.org/licenses/>.
 
17
 */
 
18
 
 
19
#include <grub/mm.h>
 
20
#include <grub/misc.h>
 
21
 
 
22
#include <grub/types.h>
 
23
#include <grub/types.h>
 
24
#include <grub/err.h>
 
25
#include <grub/cache.h>
 
26
 
 
27
#include <grub/mips/relocator.h>
 
28
#include <grub/relocator_private.h>
 
29
 
 
30
/* Do we need mips64? */
 
31
 
 
32
extern grub_uint8_t grub_relocator_forward_start;
 
33
extern grub_uint8_t grub_relocator_forward_end;
 
34
extern grub_uint8_t grub_relocator_backward_start;
 
35
extern grub_uint8_t grub_relocator_backward_end;
 
36
 
 
37
#define REGW_SIZEOF (2 * sizeof (grub_uint32_t))
 
38
#define JUMP_SIZEOF (2 * sizeof (grub_uint32_t))
 
39
 
 
40
#define RELOCATOR_SRC_SIZEOF(x) (&grub_relocator_##x##_end \
 
41
                                 - &grub_relocator_##x##_start)
 
42
#define RELOCATOR_SIZEOF(x)     (RELOCATOR_SRC_SIZEOF(x) \
 
43
                                 + REGW_SIZEOF * 3)
 
44
grub_size_t grub_relocator_align = sizeof (grub_uint32_t);
 
45
grub_size_t grub_relocator_forward_size;
 
46
grub_size_t grub_relocator_backward_size;
 
47
grub_size_t grub_relocator_jumper_size = JUMP_SIZEOF + REGW_SIZEOF;
 
48
 
 
49
void
 
50
grub_cpu_relocator_init (void)
 
51
{
 
52
  grub_relocator_forward_size = RELOCATOR_SIZEOF(forward);
 
53
  grub_relocator_backward_size = RELOCATOR_SIZEOF(backward);
 
54
}
 
55
 
 
56
static void
 
57
write_reg (int regn, grub_uint32_t val, void **target)
 
58
{
 
59
  /* lui $r, (val+0x8000).  */
 
60
  *(grub_uint32_t *) *target = ((0x3c00 | regn) << 16) | ((val + 0x8000) >> 16);
 
61
  *target = ((grub_uint32_t *) *target) + 1;
 
62
  /* addiu $r, $r, val. */
 
63
  *(grub_uint32_t *) *target = (((0x2400 | regn << 5 | regn) << 16)
 
64
                                | (val & 0xffff));
 
65
  *target = ((grub_uint32_t *) *target) + 1;
 
66
}
 
67
 
 
68
static void
 
69
write_jump (int regn, void **target)
 
70
{
 
71
  /* j $r.  */
 
72
  *(grub_uint32_t *) *target = (regn<<21) | 0x8;
 
73
  *target = ((grub_uint32_t *) *target) + 1;
 
74
  /* nop.  */
 
75
  *(grub_uint32_t *) *target = 0;
 
76
  *target = ((grub_uint32_t *) *target) + 1;
 
77
}
 
78
 
 
79
void
 
80
grub_cpu_relocator_jumper (void *rels, grub_addr_t addr)
 
81
{
 
82
  write_reg (1, addr, &rels);
 
83
  write_jump (1, &rels);
 
84
}
 
85
 
 
86
void
 
87
grub_cpu_relocator_backward (void *ptr0, void *src, void *dest,
 
88
                             grub_size_t size)
 
89
{
 
90
  void *ptr = ptr0;
 
91
  write_reg (8, (grub_uint32_t) src, &ptr);
 
92
  write_reg (9, (grub_uint32_t) dest, &ptr);
 
93
  write_reg (10, (grub_uint32_t) size, &ptr);
 
94
  grub_memcpy (ptr, &grub_relocator_backward_start,
 
95
               RELOCATOR_SRC_SIZEOF (backward));
 
96
}
 
97
 
 
98
void
 
99
grub_cpu_relocator_forward (void *ptr0, void *src, void *dest,
 
100
                             grub_size_t size)
 
101
{
 
102
  void *ptr = ptr0;
 
103
  write_reg (8, (grub_uint32_t) src, &ptr);
 
104
  write_reg (9, (grub_uint32_t) dest, &ptr);
 
105
  write_reg (10, (grub_uint32_t) size, &ptr);
 
106
  grub_memcpy (ptr, &grub_relocator_forward_start, 
 
107
               RELOCATOR_SRC_SIZEOF (forward));
 
108
}
 
109
 
 
110
grub_err_t
 
111
grub_relocator32_boot (struct grub_relocator *rel,
 
112
                       struct grub_relocator32_state state)
 
113
{
 
114
  grub_relocator_chunk_t ch;
 
115
  void *ptr;
 
116
  grub_err_t err;
 
117
  void *relst;
 
118
  grub_size_t relsize;
 
119
  grub_size_t stateset_size = 31 * REGW_SIZEOF + JUMP_SIZEOF;
 
120
  unsigned i;
 
121
  grub_addr_t vtarget;
 
122
 
 
123
  err = grub_relocator_alloc_chunk_align (rel, &ch, 0,
 
124
                                          (0xffffffff - stateset_size)
 
125
                                          + 1, stateset_size,
 
126
                                          sizeof (grub_uint32_t),
 
127
                                          GRUB_RELOCATOR_PREFERENCE_NONE);
 
128
  if (err)
 
129
    return err;
 
130
 
 
131
  ptr = get_virtual_current_address (ch);
 
132
  for (i = 1; i < 32; i++)
 
133
    write_reg (i, state.gpr[i], &ptr);
 
134
  write_jump (state.jumpreg, &ptr);
 
135
 
 
136
  vtarget = (grub_addr_t) grub_map_memory (get_physical_target_address (ch),
 
137
                                           stateset_size);
 
138
 
 
139
  err = grub_relocator_prepare_relocs (rel, vtarget, &relst, &relsize);
 
140
  if (err)
 
141
    return err;
 
142
 
 
143
  grub_arch_sync_caches ((void *) relst, relsize);
 
144
 
 
145
  ((void (*) (void)) relst) ();
 
146
 
 
147
  /* Not reached.  */
 
148
  return GRUB_ERR_NONE;
 
149
}