2009-06-08 Oliver Henshaw <oliver.henshaw@gmail.com>
[grub2/bean.git] / kern / dl.c
blob84c465ccba39bd886544a26fcfd1aaaadab5f8f8
1 /* dl.c - loadable module support */
2 /*
3 * GRUB -- GRand Unified Bootloader
4 * Copyright (C) 2002,2003,2004,2005,2007,2008 Free Software Foundation, Inc.
6 * GRUB is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
11 * GRUB is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with GRUB. If not, see <http://www.gnu.org/licenses/>.
20 #include <config.h>
21 #include <grub/elf.h>
22 #include <grub/dl.h>
23 #include <grub/misc.h>
24 #include <grub/mm.h>
25 #include <grub/err.h>
26 #include <grub/types.h>
27 #include <grub/symbol.h>
28 #include <grub/file.h>
29 #include <grub/env.h>
30 #include <grub/cache.h>
32 #if GRUB_CPU_SIZEOF_VOID_P == 4
34 typedef Elf32_Word Elf_Word;
35 typedef Elf32_Addr Elf_Addr;
36 typedef Elf32_Ehdr Elf_Ehdr;
37 typedef Elf32_Shdr Elf_Shdr;
38 typedef Elf32_Sym Elf_Sym;
40 # define ELF_ST_BIND(val) ELF32_ST_BIND (val)
41 # define ELF_ST_TYPE(val) ELF32_ST_TYPE (val)
43 #elif GRUB_CPU_SIZEOF_VOID_P == 8
45 typedef Elf64_Word Elf_Word;
46 typedef Elf64_Addr Elf_Addr;
47 typedef Elf64_Ehdr Elf_Ehdr;
48 typedef Elf64_Shdr Elf_Shdr;
49 typedef Elf64_Sym Elf_Sym;
51 # define ELF_ST_BIND(val) ELF64_ST_BIND (val)
52 # define ELF_ST_TYPE(val) ELF64_ST_TYPE (val)
54 #endif
58 struct grub_dl_list
60 struct grub_dl_list *next;
61 grub_dl_t mod;
63 typedef struct grub_dl_list *grub_dl_list_t;
65 static grub_dl_list_t grub_dl_head;
67 static grub_err_t
68 grub_dl_add (grub_dl_t mod)
70 grub_dl_list_t l;
72 if (grub_dl_get (mod->name))
73 return grub_error (GRUB_ERR_BAD_MODULE,
74 "`%s' is already loaded", mod->name);
76 l = (grub_dl_list_t) grub_malloc (sizeof (*l));
77 if (! l)
78 return grub_errno;
80 l->mod = mod;
81 l->next = grub_dl_head;
82 grub_dl_head = l;
84 return GRUB_ERR_NONE;
87 static void
88 grub_dl_remove (grub_dl_t mod)
90 grub_dl_list_t *p, q;
92 for (p = &grub_dl_head, q = *p; q; p = &q->next, q = *p)
93 if (q->mod == mod)
95 *p = q->next;
96 grub_free (q);
97 return;
101 grub_dl_t
102 grub_dl_get (const char *name)
104 grub_dl_list_t l;
106 for (l = grub_dl_head; l; l = l->next)
107 if (grub_strcmp (name, l->mod->name) == 0)
108 return l->mod;
110 return 0;
113 void
114 grub_dl_iterate (int (*hook) (grub_dl_t mod))
116 grub_dl_list_t l;
118 for (l = grub_dl_head; l; l = l->next)
119 if (hook (l->mod))
120 break;
125 struct grub_symbol
127 struct grub_symbol *next;
128 const char *name;
129 void *addr;
130 grub_dl_t mod; /* The module to which this symbol belongs. */
132 typedef struct grub_symbol *grub_symbol_t;
134 /* The size of the symbol table. */
135 #define GRUB_SYMTAB_SIZE 509
137 /* The symbol table (using an open-hash). */
138 static struct grub_symbol *grub_symtab[GRUB_SYMTAB_SIZE];
140 /* Simple hash function. */
141 static unsigned
142 grub_symbol_hash (const char *s)
144 unsigned key = 0;
146 while (*s)
147 key = key * 65599 + *s++;
149 return (key + (key >> 5)) % GRUB_SYMTAB_SIZE;
152 /* Resolve the symbol name NAME and return the address.
153 Return NULL, if not found. */
154 void *
155 grub_dl_resolve_symbol (const char *name)
157 grub_symbol_t sym;
159 for (sym = grub_symtab[grub_symbol_hash (name)]; sym; sym = sym->next)
160 if (grub_strcmp (sym->name, name) == 0)
161 return sym->addr;
163 return 0;
166 /* Register a symbol with the name NAME and the address ADDR. */
167 grub_err_t
168 grub_dl_register_symbol (const char *name, void *addr, grub_dl_t mod)
170 grub_symbol_t sym;
171 unsigned k;
173 sym = (grub_symbol_t) grub_malloc (sizeof (*sym));
174 if (! sym)
175 return grub_errno;
177 if (mod)
179 sym->name = grub_strdup (name);
180 if (! sym->name)
182 grub_free (sym);
183 return grub_errno;
186 else
187 sym->name = name;
189 sym->addr = addr;
190 sym->mod = mod;
192 k = grub_symbol_hash (name);
193 sym->next = grub_symtab[k];
194 grub_symtab[k] = sym;
196 return GRUB_ERR_NONE;
199 /* Unregister all the symbols defined in the module MOD. */
200 static void
201 grub_dl_unregister_symbols (grub_dl_t mod)
203 unsigned i;
205 if (! mod)
206 grub_fatal ("core symbols cannot be unregistered");
208 for (i = 0; i < GRUB_SYMTAB_SIZE; i++)
210 grub_symbol_t sym, *p, q;
212 for (p = &grub_symtab[i], sym = *p; sym; sym = q)
214 q = sym->next;
215 if (sym->mod == mod)
217 *p = q;
218 grub_free ((void *) sym->name);
219 grub_free (sym);
221 else
222 p = &sym->next;
227 /* Return the address of a section whose index is N. */
228 static void *
229 grub_dl_get_section_addr (grub_dl_t mod, unsigned n)
231 grub_dl_segment_t seg;
233 for (seg = mod->segment; seg; seg = seg->next)
234 if (seg->section == n)
235 return seg->addr;
237 return 0;
240 /* Check if EHDR is a valid ELF header. */
241 grub_err_t
242 grub_dl_check_header (void *ehdr, grub_size_t size)
244 Elf_Ehdr *e = ehdr;
246 /* Check the header size. */
247 if (size < sizeof (Elf_Ehdr))
248 return grub_error (GRUB_ERR_BAD_OS, "ELF header smaller than expected");
250 /* Check the magic numbers. */
251 if (grub_arch_dl_check_header (ehdr)
252 || e->e_ident[EI_MAG0] != ELFMAG0
253 || e->e_ident[EI_MAG1] != ELFMAG1
254 || e->e_ident[EI_MAG2] != ELFMAG2
255 || e->e_ident[EI_MAG3] != ELFMAG3
256 || e->e_ident[EI_VERSION] != EV_CURRENT
257 || e->e_version != EV_CURRENT)
258 return grub_error (GRUB_ERR_BAD_OS, "invalid arch independent ELF magic");
260 return GRUB_ERR_NONE;
263 /* Load all segments from memory specified by E. */
264 static grub_err_t
265 grub_dl_load_segments (grub_dl_t mod, const Elf_Ehdr *e)
267 unsigned i;
268 Elf_Shdr *s;
270 for (i = 0, s = (Elf_Shdr *)((char *) e + e->e_shoff);
271 i < e->e_shnum;
272 i++, s = (Elf_Shdr *)((char *) s + e->e_shentsize))
274 if (s->sh_flags & SHF_ALLOC)
276 grub_dl_segment_t seg;
278 seg = (grub_dl_segment_t) grub_malloc (sizeof (*seg));
279 if (! seg)
280 return grub_errno;
282 if (s->sh_size)
284 void *addr;
286 addr = grub_memalign (s->sh_addralign, s->sh_size);
287 if (! addr)
289 grub_free (seg);
290 return grub_errno;
293 switch (s->sh_type)
295 case SHT_PROGBITS:
296 grub_memcpy (addr, (char *) e + s->sh_offset, s->sh_size);
297 break;
298 case SHT_NOBITS:
299 grub_memset (addr, 0, s->sh_size);
300 break;
303 seg->addr = addr;
305 else
306 seg->addr = 0;
308 seg->size = s->sh_size;
309 seg->section = i;
310 seg->next = mod->segment;
311 mod->segment = seg;
315 return GRUB_ERR_NONE;
318 static grub_err_t
319 grub_dl_resolve_symbols (grub_dl_t mod, Elf_Ehdr *e)
321 unsigned i;
322 Elf_Shdr *s;
323 Elf_Sym *sym;
324 const char *str;
325 Elf_Word size, entsize;
327 for (i = 0, s = (Elf_Shdr *) ((char *) e + e->e_shoff);
328 i < e->e_shnum;
329 i++, s = (Elf_Shdr *) ((char *) s + e->e_shentsize))
330 if (s->sh_type == SHT_SYMTAB)
331 break;
333 if (i == e->e_shnum)
334 return grub_error (GRUB_ERR_BAD_MODULE, "no symbol table");
336 sym = (Elf_Sym *) ((char *) e + s->sh_offset);
337 size = s->sh_size;
338 entsize = s->sh_entsize;
340 s = (Elf_Shdr *) ((char *) e + e->e_shoff + e->e_shentsize * s->sh_link);
341 str = (char *) e + s->sh_offset;
343 for (i = 0;
344 i < size / entsize;
345 i++, sym = (Elf_Sym *) ((char *) sym + entsize))
347 unsigned char type = ELF_ST_TYPE (sym->st_info);
348 unsigned char bind = ELF_ST_BIND (sym->st_info);
349 const char *name = str + sym->st_name;
351 switch (type)
353 case STT_NOTYPE:
354 /* Resolve a global symbol. */
355 if (sym->st_name != 0 && sym->st_shndx == 0)
357 sym->st_value = (Elf_Addr) grub_dl_resolve_symbol (name);
358 if (! sym->st_value)
359 return grub_error (GRUB_ERR_BAD_MODULE,
360 "the symbol `%s' not found", name);
362 else
363 sym->st_value = 0;
364 break;
366 case STT_OBJECT:
367 sym->st_value += (Elf_Addr) grub_dl_get_section_addr (mod,
368 sym->st_shndx);
369 if (bind != STB_LOCAL)
370 if (grub_dl_register_symbol (name, (void *) sym->st_value, mod))
371 return grub_errno;
372 break;
374 case STT_FUNC:
375 sym->st_value += (Elf_Addr) grub_dl_get_section_addr (mod,
376 sym->st_shndx);
377 if (bind != STB_LOCAL)
378 if (grub_dl_register_symbol (name, (void *) sym->st_value, mod))
379 return grub_errno;
381 if (grub_strcmp (name, "grub_mod_init") == 0)
382 mod->init = (void (*) (grub_dl_t)) sym->st_value;
383 else if (grub_strcmp (name, "grub_mod_fini") == 0)
384 mod->fini = (void (*) (void)) sym->st_value;
385 break;
387 case STT_SECTION:
388 sym->st_value = (Elf_Addr) grub_dl_get_section_addr (mod,
389 sym->st_shndx);
390 break;
392 case STT_FILE:
393 sym->st_value = 0;
394 break;
396 default:
397 return grub_error (GRUB_ERR_BAD_MODULE,
398 "unknown symbol type `%d'", (int) type);
402 return GRUB_ERR_NONE;
405 static void
406 grub_dl_call_init (grub_dl_t mod)
408 if (mod->init)
409 (mod->init) (mod);
412 static grub_err_t
413 grub_dl_resolve_name (grub_dl_t mod, Elf_Ehdr *e)
415 Elf_Shdr *s;
416 const char *str;
417 unsigned i;
419 s = (Elf_Shdr *) ((char *) e + e->e_shoff + e->e_shstrndx * e->e_shentsize);
420 str = (char *) e + s->sh_offset;
422 for (i = 0, s = (Elf_Shdr *) ((char *) e + e->e_shoff);
423 i < e->e_shnum;
424 i++, s = (Elf_Shdr *) ((char *) s + e->e_shentsize))
425 if (grub_strcmp (str + s->sh_name, ".modname") == 0)
427 mod->name = grub_strdup ((char *) e + s->sh_offset);
428 if (! mod->name)
429 return grub_errno;
430 break;
433 if (i == e->e_shnum)
434 return grub_error (GRUB_ERR_BAD_MODULE, "no module name found");
436 return GRUB_ERR_NONE;
439 static grub_err_t
440 grub_dl_resolve_dependencies (grub_dl_t mod, Elf_Ehdr *e)
442 Elf_Shdr *s;
443 const char *str;
444 unsigned i;
446 s = (Elf_Shdr *) ((char *) e + e->e_shoff + e->e_shstrndx * e->e_shentsize);
447 str = (char *) e + s->sh_offset;
449 for (i = 0, s = (Elf_Shdr *) ((char *) e + e->e_shoff);
450 i < e->e_shnum;
451 i++, s = (Elf_Shdr *) ((char *) s + e->e_shentsize))
452 if (grub_strcmp (str + s->sh_name, ".moddeps") == 0)
454 const char *name = (char *) e + s->sh_offset;
455 const char *max = name + s->sh_size;
457 while ((name < max) && (*name))
459 grub_dl_t m;
460 grub_dl_dep_t dep;
462 m = grub_dl_load (name);
463 if (! m)
464 return grub_errno;
466 grub_dl_ref (m);
468 dep = (grub_dl_dep_t) grub_malloc (sizeof (*dep));
469 if (! dep)
470 return grub_errno;
472 dep->mod = m;
473 dep->next = mod->dep;
474 mod->dep = dep;
476 name += grub_strlen (name) + 1;
480 return GRUB_ERR_NONE;
483 #ifndef GRUB_UTIL
485 grub_dl_ref (grub_dl_t mod)
487 grub_dl_dep_t dep;
489 for (dep = mod->dep; dep; dep = dep->next)
490 grub_dl_ref (dep->mod);
492 return ++mod->ref_count;
496 grub_dl_unref (grub_dl_t mod)
498 grub_dl_dep_t dep;
500 for (dep = mod->dep; dep; dep = dep->next)
501 grub_dl_unref (dep->mod);
503 return --mod->ref_count;
505 #endif
507 static void
508 grub_dl_flush_cache (grub_dl_t mod)
510 grub_dl_segment_t seg;
512 for (seg = mod->segment; seg; seg = seg->next) {
513 if (seg->size) {
514 grub_dprintf ("modules", "flushing 0x%lx bytes at %p\n",
515 (unsigned long) seg->size, seg->addr);
516 grub_arch_sync_caches (seg->addr, seg->size);
521 /* Load a module from core memory. */
522 grub_dl_t
523 grub_dl_load_core (void *addr, grub_size_t size)
525 Elf_Ehdr *e;
526 grub_dl_t mod;
528 grub_dprintf ("modules", "module at %p, size 0x%lx\n", addr,
529 (unsigned long) size);
530 e = addr;
531 if (grub_dl_check_header (e, size))
532 return 0;
534 if (e->e_type != ET_REL)
536 grub_error (GRUB_ERR_BAD_MODULE, "invalid ELF file type");
537 return 0;
540 /* Make sure that every section is within the core. */
541 if (size < e->e_shoff + e->e_shentsize * e->e_shnum)
543 grub_error (GRUB_ERR_BAD_OS, "ELF sections outside core");
544 return 0;
547 mod = (grub_dl_t) grub_malloc (sizeof (*mod));
548 if (! mod)
549 return 0;
551 mod->name = 0;
552 mod->ref_count = 1;
553 mod->dep = 0;
554 mod->segment = 0;
555 mod->init = 0;
556 mod->fini = 0;
558 grub_dprintf ("modules", "relocating to %p\n", mod);
559 if (grub_dl_resolve_name (mod, e)
560 || grub_dl_resolve_dependencies (mod, e)
561 || grub_dl_load_segments (mod, e)
562 || grub_dl_resolve_symbols (mod, e)
563 || grub_arch_dl_relocate_symbols (mod, e))
565 mod->fini = 0;
566 grub_dl_unload (mod);
567 return 0;
570 grub_dl_flush_cache (mod);
572 grub_dprintf ("modules", "module name: %s\n", mod->name);
573 grub_dprintf ("modules", "init function: %p\n", mod->init);
574 grub_dl_call_init (mod);
576 if (grub_dl_add (mod))
578 grub_dl_unload (mod);
579 return 0;
582 return mod;
585 /* Load a module from the file FILENAME. */
586 grub_dl_t
587 grub_dl_load_file (const char *filename)
589 grub_file_t file = NULL;
590 grub_ssize_t size;
591 void *core = 0;
592 grub_dl_t mod = 0;
594 file = grub_file_open (filename);
595 if (! file)
596 return 0;
598 size = grub_file_size (file);
599 core = grub_malloc (size);
600 if (! core)
602 grub_file_close (file);
603 return 0;
606 if (grub_file_read (file, core, size) != (int) size)
608 grub_file_close (file);
609 grub_free (core);
610 return 0;
613 /* We must close this before we try to process dependencies.
614 Some disk backends do not handle gracefully multiple concurrent
615 opens of the same device. */
616 grub_file_close (file);
618 mod = grub_dl_load_core (core, size);
619 if (! mod)
621 grub_free (core);
622 return 0;
625 mod->ref_count = 0;
626 return mod;
629 /* Load a module using a symbolic name. */
630 grub_dl_t
631 grub_dl_load (const char *name)
633 char *filename;
634 grub_dl_t mod;
635 char *grub_dl_dir = grub_env_get ("prefix");
637 mod = grub_dl_get (name);
638 if (mod)
639 return mod;
641 if (! grub_dl_dir) {
642 grub_error (GRUB_ERR_FILE_NOT_FOUND, "\"prefix\" is not set");
643 return 0;
646 filename = (char *) grub_malloc (grub_strlen (grub_dl_dir) + 1
647 + grub_strlen (name) + 4 + 1);
648 if (! filename)
649 return 0;
651 grub_sprintf (filename, "%s/%s.mod", grub_dl_dir, name);
652 mod = grub_dl_load_file (filename);
653 grub_free (filename);
655 if (! mod)
656 return 0;
658 if (grub_strcmp (mod->name, name) != 0)
659 grub_error (GRUB_ERR_BAD_MODULE, "mismatched names");
661 return mod;
664 /* Unload the module MOD. */
666 grub_dl_unload (grub_dl_t mod)
668 grub_dl_dep_t dep, depn;
669 grub_dl_segment_t seg, segn;
671 if (mod->ref_count > 0)
672 return 0;
674 if (mod->fini)
675 (mod->fini) ();
677 grub_dl_remove (mod);
678 grub_dl_unregister_symbols (mod);
680 for (dep = mod->dep; dep; dep = depn)
682 depn = dep->next;
684 if (! grub_dl_unref (dep->mod))
685 grub_dl_unload (dep->mod);
687 grub_free (dep);
690 for (seg = mod->segment; seg; seg = segn)
692 segn = seg->next;
693 grub_free (seg->addr);
694 grub_free (seg);
697 grub_free (mod->name);
698 grub_free (mod);
699 return 1;
702 /* Unload unneeded modules. */
703 void
704 grub_dl_unload_unneeded (void)
706 /* Because grub_dl_remove modifies the list of modules, this
707 implementation is tricky. */
708 grub_dl_list_t p = grub_dl_head;
710 while (p)
712 if (grub_dl_unload (p->mod))
714 p = grub_dl_head;
715 continue;
718 p = p->next;
722 /* Unload all modules. */
723 void
724 grub_dl_unload_all (void)
726 while (grub_dl_head)
728 grub_dl_list_t p;
730 grub_dl_unload_unneeded ();
732 /* Force to decrement the ref count. This will purge pre-loaded
733 modules and manually inserted modules. */
734 for (p = grub_dl_head; p; p = p->next)
735 p->mod->ref_count--;