Initial import of Leif's work
This commit is contained in:
		
							parent
							
								
									21026747df
								
							
						
					
					
						commit
						389b31cd71
					
				
					 65 changed files with 7311 additions and 13 deletions
				
			
		|  | @ -174,3 +174,5 @@ po/*.gmo | ||||||
| po/LINGUAS | po/LINGUAS | ||||||
| include/grub/gcrypt/gcrypt.h | include/grub/gcrypt/gcrypt.h | ||||||
| include/grub/gcrypt/g10lib.h | include/grub/gcrypt/g10lib.h | ||||||
|  | grub-core/lib/dtc-grub | ||||||
|  | grub-core/Makefile.libfdt.def | ||||||
|  |  | ||||||
|  | @ -150,6 +150,8 @@ program = { | ||||||
|   common = util/resolve.c; |   common = util/resolve.c; | ||||||
|   common = grub-core/kern/emu/argp_common.c; |   common = grub-core/kern/emu/argp_common.c; | ||||||
| 
 | 
 | ||||||
|  |   arm = grub-core/kern/arm/dl.c; | ||||||
|  | 
 | ||||||
|   extra_dist = util/grub-mkimagexx.c; |   extra_dist = util/grub-mkimagexx.c; | ||||||
| 
 | 
 | ||||||
|   ldadd = libgrubmods.a; |   ldadd = libgrubmods.a; | ||||||
|  | @ -470,6 +472,7 @@ script = { | ||||||
|   enable = mips_loongson; |   enable = mips_loongson; | ||||||
|   enable = ia64_efi; |   enable = ia64_efi; | ||||||
|   enable = powerpc_ieee1275; |   enable = powerpc_ieee1275; | ||||||
|  |   enable = arm_uboot; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| script = { | script = { | ||||||
|  |  | ||||||
|  | @ -30,6 +30,9 @@ for x in mpi-asm-defs.h mpih-add1.c mpih-sub1.c mpih-mul1.c mpih-mul2.c mpih-mul | ||||||
|     ln -s generic/"$x" grub-core/lib/libgcrypt-grub/mpi/"$x" |     ln -s generic/"$x" grub-core/lib/libgcrypt-grub/mpi/"$x" | ||||||
| done | done | ||||||
| 
 | 
 | ||||||
|  | echo "Importing libfdt..." | ||||||
|  | python util/import_libfdt.py grub-core/lib/dtc/ grub-core | ||||||
|  | 
 | ||||||
| echo "Creating Makefile.tpl..." | echo "Creating Makefile.tpl..." | ||||||
| python gentpl.py | sed -e '/^$/{N;/^\n$/D;}' > Makefile.tpl | python gentpl.py | sed -e '/^$/{N;/^\n$/D;}' > Makefile.tpl | ||||||
| 
 | 
 | ||||||
|  | @ -43,7 +46,7 @@ if [ "x${GRUB_CONTRIB}" != x ]; then | ||||||
| fi | fi | ||||||
| 
 | 
 | ||||||
| UTIL_DEFS='Makefile.util.def Makefile.utilgcry.def' | UTIL_DEFS='Makefile.util.def Makefile.utilgcry.def' | ||||||
| CORE_DEFS='grub-core/Makefile.core.def grub-core/Makefile.gcry.def' | CORE_DEFS='grub-core/Makefile.core.def grub-core/Makefile.gcry.def grub-core/Makefile.libfdt.def' | ||||||
| 
 | 
 | ||||||
| for extra in contrib/*/Makefile.util.def; do | for extra in contrib/*/Makefile.util.def; do | ||||||
|   if test -e "$extra"; then |   if test -e "$extra"; then | ||||||
|  |  | ||||||
|  | @ -37,6 +37,13 @@ if COND_sparc64_ieee1275 | ||||||
|   CFLAGS_PLATFORM += -mno-app-regs |   CFLAGS_PLATFORM += -mno-app-regs | ||||||
|   LDFLAGS_PLATFORM = -Wl,-melf64_sparc -mno-relax |   LDFLAGS_PLATFORM = -Wl,-melf64_sparc -mno-relax | ||||||
| endif | endif | ||||||
|  | if COND_arm | ||||||
|  | # Image entry point always in ARM (A32) state - ensure proper functionality if | ||||||
|  | # the rest is built for the Thumb (T32) state. | ||||||
|  |   CFLAGS_PLATFORM += -mthumb-interwork -mno-unaligned-access -mlong-calls | ||||||
|  |   CCASFLAGS_PLATFORM = -Wa,-mimplicit-it=thumb | ||||||
|  |   LDFLAGS_PLATFORM = -Wl,--wrap=__clear_cache | ||||||
|  | endif | ||||||
| 
 | 
 | ||||||
| # Other options | # Other options | ||||||
| 
 | 
 | ||||||
|  | @ -110,6 +117,8 @@ CPPFLAGS_GNULIB = -I$(top_builddir)/grub-core/gnulib -I$(top_srcdir)/grub-core/g | ||||||
| CFLAGS_POSIX = -fno-builtin | CFLAGS_POSIX = -fno-builtin | ||||||
| CPPFLAGS_POSIX = -I$(top_srcdir)/grub-core/lib/posix_wrap | CPPFLAGS_POSIX = -I$(top_srcdir)/grub-core/lib/posix_wrap | ||||||
| 
 | 
 | ||||||
|  | CPPFLAGS_LIBFDT = -I$(top_srcdir)/grub-core/lib/dtc-grub/libfdt $(CPPFLAGS_POSIX) | ||||||
|  | 
 | ||||||
| CFLAGS_GCRY = -Wno-error -Wno-missing-field-initializers $(CFLAGS_POSIX) | CFLAGS_GCRY = -Wno-error -Wno-missing-field-initializers $(CFLAGS_POSIX) | ||||||
| CPPFLAGS_GCRY = -I$(top_srcdir)/grub-core/lib/libgcrypt_wrap $(CPPFLAGS_POSIX) -D_GCRYPT_IN_LIBGCRYPT=1 -I$(top_srcdir)/include/grub/gcrypt | CPPFLAGS_GCRY = -I$(top_srcdir)/grub-core/lib/libgcrypt_wrap $(CPPFLAGS_POSIX) -D_GCRYPT_IN_LIBGCRYPT=1 -I$(top_srcdir)/include/grub/gcrypt | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -18,6 +18,7 @@ EXTRA_DIST += conf/i386-pc-cygwin-img-ld.sc | ||||||
| 
 | 
 | ||||||
| EXTRA_DIST += grub-core/Makefile.core.def | EXTRA_DIST += grub-core/Makefile.core.def | ||||||
| EXTRA_DIST += grub-core/Makefile.gcry.def | EXTRA_DIST += grub-core/Makefile.gcry.def | ||||||
|  | EXTRA_DIST += grub-core/Makefile.libfdt.def | ||||||
| 
 | 
 | ||||||
| EXTRA_DIST += grub-core/genmoddep.awk | EXTRA_DIST += grub-core/genmoddep.awk | ||||||
| EXTRA_DIST += grub-core/genmod.sh.in | EXTRA_DIST += grub-core/genmod.sh.in | ||||||
|  | @ -27,6 +28,7 @@ EXTRA_DIST += grub-core/genemuinit.sh | ||||||
| EXTRA_DIST += grub-core/genemuinitheader.sh | EXTRA_DIST += grub-core/genemuinitheader.sh | ||||||
| 
 | 
 | ||||||
| EXTRA_DIST += grub-core/lib/libgcrypt/cipher | EXTRA_DIST += grub-core/lib/libgcrypt/cipher | ||||||
|  | EXTRA_DIST += grub-core/lib/dtc | ||||||
| EXTRA_DIST += $(shell find $(top_srcdir)/include -name '*.h') | EXTRA_DIST += $(shell find $(top_srcdir)/include -name '*.h') | ||||||
| EXTRA_DIST += $(shell find $(top_srcdir)/grub-core/lib -name '*.h') | EXTRA_DIST += $(shell find $(top_srcdir)/grub-core/lib -name '*.h') | ||||||
| EXTRA_DIST += $(shell find $(top_srcdir)/grub-core/gnulib -name '*.h') | EXTRA_DIST += $(shell find $(top_srcdir)/grub-core/gnulib -name '*.h') | ||||||
|  |  | ||||||
							
								
								
									
										11
									
								
								configure.ac
									
										
									
									
									
								
							
							
						
						
									
										11
									
								
								configure.ac
									
										
									
									
									
								
							|  | @ -94,6 +94,9 @@ case "$target_cpu" in | ||||||
|                 target_cpu=mips; |                 target_cpu=mips; | ||||||
| 		machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_CPU_MIPS=1"; | 		machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_CPU_MIPS=1"; | ||||||
| 		;; | 		;; | ||||||
|  |   arm*) | ||||||
|  | 		target_cpu=arm; | ||||||
|  | 		;; | ||||||
| esac | esac | ||||||
| 
 | 
 | ||||||
| # Specify the platform (such as firmware). | # Specify the platform (such as firmware). | ||||||
|  | @ -114,6 +117,7 @@ if test "x$with_platform" = x; then | ||||||
|     mipsel-*) platform=loongson ;; |     mipsel-*) platform=loongson ;; | ||||||
|     mips-*) platform=arc ;; |     mips-*) platform=arc ;; | ||||||
|     ia64-*) platform=efi ;; |     ia64-*) platform=efi ;; | ||||||
|  |     arm-*) platform=uboot ;; | ||||||
|     *) AC_MSG_ERROR([unsupported CPU: "$target_cpu"]) ;; |     *) AC_MSG_ERROR([unsupported CPU: "$target_cpu"]) ;; | ||||||
|   esac |   esac | ||||||
| else | else | ||||||
|  | @ -148,6 +152,8 @@ case "$target_cpu"-"$platform" in | ||||||
|   mipsel-yeeloong) platform=loongson ;; |   mipsel-yeeloong) platform=loongson ;; | ||||||
|   mipsel-fuloong) platform=loongson ;; |   mipsel-fuloong) platform=loongson ;; | ||||||
|   mipsel-loongson) ;; |   mipsel-loongson) ;; | ||||||
|  |   arm-uboot) ;; | ||||||
|  |   arm-efi) ;; | ||||||
|   *-emu) ;; |   *-emu) ;; | ||||||
|   *) AC_MSG_ERROR([platform "$platform" is not supported for target CPU "$target_cpu"]) ;; |   *) AC_MSG_ERROR([platform "$platform" is not supported for target CPU "$target_cpu"]) ;; | ||||||
| esac | esac | ||||||
|  | @ -179,6 +185,7 @@ case "$platform" in | ||||||
|   multiboot)	machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_MULTIBOOT=1" ;; |   multiboot)	machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_MULTIBOOT=1" ;; | ||||||
|   efi)		machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_EFI=1" ;; |   efi)		machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_EFI=1" ;; | ||||||
|   ieee1275)	machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_IEEE1275=1" ;; |   ieee1275)	machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_IEEE1275=1" ;; | ||||||
|  |   uboot)	machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_UBOOT=1" ;; | ||||||
|   qemu)		machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_QEMU=1" ;; |   qemu)		machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_QEMU=1" ;; | ||||||
|   pc)		machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_PCBIOS=1" ;; |   pc)		machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_PCBIOS=1" ;; | ||||||
|   emu)		machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_EMU=1" ;; |   emu)		machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_EMU=1" ;; | ||||||
|  | @ -187,6 +194,7 @@ case "$platform" in | ||||||
|   arc)	machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_ARC=1" ;; |   arc)	machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_ARC=1" ;; | ||||||
| esac | esac | ||||||
| case "$target_cpu" in | case "$target_cpu" in | ||||||
|  |   arm)      	machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_ARM=1" ;; | ||||||
|   mips |mipsel)      	machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_MIPS=1" ;; |   mips |mipsel)      	machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_MIPS=1" ;; | ||||||
|   sparc64)      machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_SPARC64=1" ;; |   sparc64)      machine_CPPFLAGS="$machine_CPPFLAGS -DGRUB_MACHINE_SPARC64=1" ;; | ||||||
| esac | esac | ||||||
|  | @ -1150,6 +1158,9 @@ AM_CONDITIONAL([COND_powerpc_ieee1275], [test x$target_cpu = xpowerpc -a x$platf | ||||||
| AM_CONDITIONAL([COND_mips], [test x$target_cpu = xmips -o x$target_cpu = xmipsel]) | AM_CONDITIONAL([COND_mips], [test x$target_cpu = xmips -o x$target_cpu = xmipsel]) | ||||||
| AM_CONDITIONAL([COND_mipsel], [test x$target_cpu = xmipsel]) | AM_CONDITIONAL([COND_mipsel], [test x$target_cpu = xmipsel]) | ||||||
| AM_CONDITIONAL([COND_mipseb], [test x$target_cpu = xmips]) | AM_CONDITIONAL([COND_mipseb], [test x$target_cpu = xmips]) | ||||||
|  | AM_CONDITIONAL([COND_arm], [test x$target_cpu = xarm ]) | ||||||
|  | AM_CONDITIONAL([COND_arm_uboot], [test x$target_cpu = xarm -a x$platform = xuboot]) | ||||||
|  | AM_CONDITIONAL([COND_arm_efi], [test x$target_cpu = xarm -a x$platform = xefi]) | ||||||
| 
 | 
 | ||||||
| AM_CONDITIONAL([COND_HOST_HURD], [test x$host_kernel = xhurd]) | AM_CONDITIONAL([COND_HOST_HURD], [test x$host_kernel = xhurd]) | ||||||
| AM_CONDITIONAL([COND_HOST_LINUX], [test x$host_kernel = xlinux]) | AM_CONDITIONAL([COND_HOST_LINUX], [test x$host_kernel = xlinux]) | ||||||
|  |  | ||||||
|  | @ -3389,6 +3389,7 @@ you forget a command, you can run the command @command{help} | ||||||
| * crc::                         Compute or check CRC32 checksums | * crc::                         Compute or check CRC32 checksums | ||||||
| * cryptomount::                 Mount a crypto device | * cryptomount::                 Mount a crypto device | ||||||
| * date::                        Display or set current date and time | * date::                        Display or set current date and time | ||||||
|  | * devicetree::                  Load a device tree blob | ||||||
| * drivemap::                    Map a drive to another | * drivemap::                    Map a drive to another | ||||||
| * echo::                        Display a line of text | * echo::                        Display a line of text | ||||||
| * export::                      Export an environment variable | * export::                      Export an environment variable | ||||||
|  | @ -3698,6 +3699,17 @@ hour, minute, and second unchanged. | ||||||
| @end deffn | @end deffn | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @node devicetree | ||||||
|  | @subsection linux | ||||||
|  | 
 | ||||||
|  | @deffn Command devicetree file | ||||||
|  | Load a device tree blob (.dtb) from a filesystem, for later use by a Linux | ||||||
|  | kernel. Does not perform merging with any device tree supplied by firmware, | ||||||
|  | but rather replaces it completely. | ||||||
|  | @ref{GNU/Linux}. | ||||||
|  | @end deffn | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| @node drivemap | @node drivemap | ||||||
| @subsection drivemap | @subsection drivemap | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										11
									
								
								gentpl.py
									
										
									
									
									
								
							
							
						
						
									
										11
									
								
								gentpl.py
									
										
									
									
									
								
							|  | @ -23,7 +23,7 @@ GRUB_PLATFORMS = [ "emu", "i386_pc", "i386_efi", "i386_qemu", "i386_coreboot", | ||||||
|                    "i386_multiboot", "i386_ieee1275", "x86_64_efi", |                    "i386_multiboot", "i386_ieee1275", "x86_64_efi", | ||||||
|                    "mips_loongson", "sparc64_ieee1275", |                    "mips_loongson", "sparc64_ieee1275", | ||||||
|                    "powerpc_ieee1275", "mips_arc", "ia64_efi", |                    "powerpc_ieee1275", "mips_arc", "ia64_efi", | ||||||
|                    "mips_qemu_mips" ] |                    "mips_qemu_mips", "arm_uboot", "arm_efi" ] | ||||||
| 
 | 
 | ||||||
| GROUPS = {} | GROUPS = {} | ||||||
| 
 | 
 | ||||||
|  | @ -36,10 +36,12 @@ GROUPS["x86"]      = GROUPS["i386"] + GROUPS["x86_64"] | ||||||
| GROUPS["mips"]     = [ "mips_loongson", "mips_qemu_mips", "mips_arc" ] | GROUPS["mips"]     = [ "mips_loongson", "mips_qemu_mips", "mips_arc" ] | ||||||
| GROUPS["sparc64"]  = [ "sparc64_ieee1275" ] | GROUPS["sparc64"]  = [ "sparc64_ieee1275" ] | ||||||
| GROUPS["powerpc"]  = [ "powerpc_ieee1275" ] | GROUPS["powerpc"]  = [ "powerpc_ieee1275" ] | ||||||
|  | GROUPS["arm"]      = [ "arm_uboot", "arm_efi" ] | ||||||
| 
 | 
 | ||||||
| # Groups based on firmware | # Groups based on firmware | ||||||
| GROUPS["efi"]  = [ "i386_efi", "x86_64_efi", "ia64_efi" ] | GROUPS["efi"]  = [ "i386_efi", "x86_64_efi", "ia64_efi", "arm_efi" ] | ||||||
| GROUPS["ieee1275"]   = [ "i386_ieee1275", "sparc64_ieee1275", "powerpc_ieee1275" ] | GROUPS["ieee1275"]   = [ "i386_ieee1275", "sparc64_ieee1275", "powerpc_ieee1275" ] | ||||||
|  | GROUPS["uboot"] = [ "arm_uboot" ] | ||||||
| 
 | 
 | ||||||
| # emu is a special case so many core functionality isn't needed on this platform | # emu is a special case so many core functionality isn't needed on this platform | ||||||
| GROUPS["noemu"]   = GRUB_PLATFORMS[:]; GROUPS["noemu"].remove("emu") | GROUPS["noemu"]   = GRUB_PLATFORMS[:]; GROUPS["noemu"].remove("emu") | ||||||
|  | @ -57,10 +59,13 @@ GROUPS["videomodules"]   = GRUB_PLATFORMS[:]; | ||||||
| for i in GROUPS["videoinkernel"]: GROUPS["videomodules"].remove(i) | for i in GROUPS["videoinkernel"]: GROUPS["videomodules"].remove(i) | ||||||
| 
 | 
 | ||||||
| # Similar for terminfo | # Similar for terminfo | ||||||
| GROUPS["terminfoinkernel"] = ["mips_loongson", "mips_arc", "mips_qemu_mips" ] + GROUPS["ieee1275"]; | GROUPS["terminfoinkernel"] = ["mips_loongson", "mips_arc", "mips_qemu_mips" ] + GROUPS["ieee1275"] + GROUPS["uboot"]; | ||||||
| GROUPS["terminfomodule"]   = GRUB_PLATFORMS[:]; | GROUPS["terminfomodule"]   = GRUB_PLATFORMS[:]; | ||||||
| for i in GROUPS["terminfoinkernel"]: GROUPS["terminfomodule"].remove(i) | for i in GROUPS["terminfoinkernel"]: GROUPS["terminfomodule"].remove(i) | ||||||
| 
 | 
 | ||||||
|  | # Flattened Device Trees (FDT) | ||||||
|  | GROUPS["fdt"] = [ "arm_uboot", "arm_efi" ] | ||||||
|  | 
 | ||||||
| # Miscelaneous groups schedulded to disappear in future | # Miscelaneous groups schedulded to disappear in future | ||||||
| GROUPS["i386_coreboot_multiboot_qemu"] = ["i386_coreboot", "i386_multiboot", "i386_qemu"] | GROUPS["i386_coreboot_multiboot_qemu"] = ["i386_coreboot", "i386_multiboot", "i386_qemu"] | ||||||
| GROUPS["nopc"] = GRUB_PLATFORMS[:]; GROUPS["nopc"].remove("i386_pc") | GROUPS["nopc"] = GRUB_PLATFORMS[:]; GROUPS["nopc"].remove("i386_pc") | ||||||
|  |  | ||||||
|  | @ -211,6 +211,19 @@ KERNEL_HEADER_FILES += $(top_srcdir)/include/grub/extcmd.h | ||||||
| KERNEL_HEADER_FILES += $(top_srcdir)/include/grub/lib/arg.h | KERNEL_HEADER_FILES += $(top_srcdir)/include/grub/lib/arg.h | ||||||
| endif | endif | ||||||
| 
 | 
 | ||||||
|  | if COND_arm_uboot | ||||||
|  | KERNEL_HEADER_FILES += $(top_srcdir)/include/grub/uboot/uboot.h | ||||||
|  | KERNEL_HEADER_FILES += $(top_srcdir)/include/grub/uboot/disk.h | ||||||
|  | KERNEL_HEADER_FILES += $(top_srcdir)/include/grub/extcmd.h | ||||||
|  | KERNEL_HEADER_FILES += $(top_srcdir)/include/grub/lib/arg.h | ||||||
|  | endif | ||||||
|  | 
 | ||||||
|  | if COND_arm_efi | ||||||
|  | KERNEL_HEADER_FILES += $(top_srcdir)/include/grub/arm/efi/loader.h | ||||||
|  | KERNEL_HEADER_FILES += $(top_srcdir)/include/grub/efi/efi.h | ||||||
|  | KERNEL_HEADER_FILES += $(top_srcdir)/include/grub/efi/disk.h | ||||||
|  | endif | ||||||
|  | 
 | ||||||
| if COND_emu | if COND_emu | ||||||
| KERNEL_HEADER_FILES += $(top_srcdir)/include/grub/datetime.h | KERNEL_HEADER_FILES += $(top_srcdir)/include/grub/datetime.h | ||||||
| KERNEL_HEADER_FILES += $(top_srcdir)/include/grub/emu/misc.h | KERNEL_HEADER_FILES += $(top_srcdir)/include/grub/emu/misc.h | ||||||
|  |  | ||||||
|  | @ -45,6 +45,9 @@ kernel = { | ||||||
|   ia64_efi_ldflags = '-Wl,-r,-d'; |   ia64_efi_ldflags = '-Wl,-r,-d'; | ||||||
|   ia64_efi_stripflags = '--strip-unneeded -K start -R .note -R .comment -R .note.gnu.gold-version'; |   ia64_efi_stripflags = '--strip-unneeded -K start -R .note -R .comment -R .note.gnu.gold-version'; | ||||||
| 
 | 
 | ||||||
|  |   arm_efi_ldflags          = '-Wl,-r,-d'; | ||||||
|  |   arm_efi_stripflags       = '--strip-unneeded -K start -R .note -R .comment -R .note.gnu.gold-version'; | ||||||
|  | 
 | ||||||
|   i386_pc_ldflags          = '$(TARGET_IMG_LDFLAGS)'; |   i386_pc_ldflags          = '$(TARGET_IMG_LDFLAGS)'; | ||||||
|   i386_pc_ldflags          = '$(TARGET_IMG_BASE_LDOPT),0x9000'; |   i386_pc_ldflags          = '$(TARGET_IMG_BASE_LDOPT),0x9000'; | ||||||
| 
 | 
 | ||||||
|  | @ -66,6 +69,8 @@ kernel = { | ||||||
|   i386_qemu_cppflags     = '-DGRUB_BOOT_MACHINE_LINK_ADDR=$(GRUB_BOOT_MACHINE_LINK_ADDR)'; |   i386_qemu_cppflags     = '-DGRUB_BOOT_MACHINE_LINK_ADDR=$(GRUB_BOOT_MACHINE_LINK_ADDR)'; | ||||||
|   emu_cflags = '$(CFLAGS_GNULIB)'; |   emu_cflags = '$(CFLAGS_GNULIB)'; | ||||||
|   emu_cppflags = '$(CPPFLAGS_GNULIB)'; |   emu_cppflags = '$(CPPFLAGS_GNULIB)'; | ||||||
|  |   arm_uboot_ldflags       = '-Wl,-Ttext=0x08000000'; | ||||||
|  |   arm_uboot_stripflags    = '--strip-unneeded -K start -R .note -R .comment -R .note.gnu.gold-version'; | ||||||
| 
 | 
 | ||||||
|   i386_pc_startup = kern/i386/pc/startup.S; |   i386_pc_startup = kern/i386/pc/startup.S; | ||||||
|   i386_efi_startup = kern/i386/efi/startup.S; |   i386_efi_startup = kern/i386/efi/startup.S; | ||||||
|  | @ -77,6 +82,8 @@ kernel = { | ||||||
|   mips_startup = kern/mips/startup.S; |   mips_startup = kern/mips/startup.S; | ||||||
|   sparc64_ieee1275_startup = kern/sparc64/ieee1275/crt0.S; |   sparc64_ieee1275_startup = kern/sparc64/ieee1275/crt0.S; | ||||||
|   powerpc_ieee1275_startup = kern/powerpc/ieee1275/startup.S; |   powerpc_ieee1275_startup = kern/powerpc/ieee1275/startup.S; | ||||||
|  |   arm_uboot_startup = kern/arm/uboot/startup.S; | ||||||
|  |   arm_efi_startup = kern/arm/efi/startup.S; | ||||||
| 
 | 
 | ||||||
|   common = kern/command.c; |   common = kern/command.c; | ||||||
|   common = kern/corecmd.c; |   common = kern/corecmd.c; | ||||||
|  | @ -112,6 +119,12 @@ kernel = { | ||||||
|   ieee1275 = term/ieee1275/console.c; |   ieee1275 = term/ieee1275/console.c; | ||||||
|   ieee1275 = kern/ieee1275/init.c; |   ieee1275 = kern/ieee1275/init.c; | ||||||
| 
 | 
 | ||||||
|  |   uboot = disk/uboot/ubootdisk.c; | ||||||
|  |   uboot = kern/uboot/uboot.c; | ||||||
|  |   uboot = kern/uboot/init.c; | ||||||
|  |   uboot = kern/uboot/hw.c; | ||||||
|  |   uboot = term/uboot/console.c; | ||||||
|  | 
 | ||||||
|   terminfoinkernel = term/terminfo.c; |   terminfoinkernel = term/terminfo.c; | ||||||
|   terminfoinkernel = term/tparm.c; |   terminfoinkernel = term/tparm.c; | ||||||
|   terminfoinkernel = commands/extcmd.c; |   terminfoinkernel = commands/extcmd.c; | ||||||
|  | @ -143,6 +156,9 @@ kernel = { | ||||||
|   ia64_efi = kern/ia64/dl.c; |   ia64_efi = kern/ia64/dl.c; | ||||||
|   ia64_efi = kern/ia64/dl_helper.c; |   ia64_efi = kern/ia64/dl_helper.c; | ||||||
| 
 | 
 | ||||||
|  |   arm_efi = kern/arm/efi/init.c; | ||||||
|  |   arm_efi = kern/arm/efi/misc.c; | ||||||
|  | 
 | ||||||
|   i386_pc = kern/i386/pc/init.c; |   i386_pc = kern/i386/pc/init.c; | ||||||
|   i386_pc = kern/i386/pc/mmap.c; |   i386_pc = kern/i386/pc/mmap.c; | ||||||
|   i386_pc = term/i386/pc/console.c; |   i386_pc = term/i386/pc/console.c; | ||||||
|  | @ -192,6 +208,10 @@ kernel = { | ||||||
|   sparc64_ieee1275 = kern/sparc64/dl.c; |   sparc64_ieee1275 = kern/sparc64/dl.c; | ||||||
|   sparc64_ieee1275 = kern/sparc64/ieee1275/ieee1275.c; |   sparc64_ieee1275 = kern/sparc64/ieee1275/ieee1275.c; | ||||||
| 
 | 
 | ||||||
|  |   arm = kern/arm/dl.c; | ||||||
|  |   arm = kern/arm/cache.S; | ||||||
|  |   arm = kern/arm/misc.S; | ||||||
|  | 
 | ||||||
|   emu = disk/host.c; |   emu = disk/host.c; | ||||||
|   emu = gnulib/progname.c; |   emu = gnulib/progname.c; | ||||||
|   emu = gnulib/error.c; |   emu = gnulib/error.c; | ||||||
|  | @ -688,6 +708,7 @@ module = { | ||||||
|   efi = lib/efi/halt.c; |   efi = lib/efi/halt.c; | ||||||
|   ieee1275 = lib/ieee1275/halt.c; |   ieee1275 = lib/ieee1275/halt.c; | ||||||
|   emu = lib/emu/halt.c; |   emu = lib/emu/halt.c; | ||||||
|  |   uboot = lib/uboot/halt.c; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| module = { | module = { | ||||||
|  | @ -696,11 +717,13 @@ module = { | ||||||
|   i386 = lib/i386/reboot_trampoline.S; |   i386 = lib/i386/reboot_trampoline.S; | ||||||
|   ia64_efi = lib/efi/reboot.c; |   ia64_efi = lib/efi/reboot.c; | ||||||
|   x86_64_efi = lib/efi/reboot.c; |   x86_64_efi = lib/efi/reboot.c; | ||||||
|  |   arm_efi = lib/efi/reboot.c; | ||||||
|   powerpc_ieee1275 = lib/ieee1275/reboot.c; |   powerpc_ieee1275 = lib/ieee1275/reboot.c; | ||||||
|   sparc64_ieee1275 = lib/ieee1275/reboot.c; |   sparc64_ieee1275 = lib/ieee1275/reboot.c; | ||||||
|   mips_arc = lib/mips/arc/reboot.c; |   mips_arc = lib/mips/arc/reboot.c; | ||||||
|   mips_loongson = lib/mips/loongson/reboot.c; |   mips_loongson = lib/mips/loongson/reboot.c; | ||||||
|   mips_qemu_mips = lib/mips/qemu_mips/reboot.c; |   mips_qemu_mips = lib/mips/qemu_mips/reboot.c; | ||||||
|  |   uboot = lib/uboot/reboot.c; | ||||||
|   common = commands/reboot.c; |   common = commands/reboot.c; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | @ -1358,6 +1381,7 @@ module = { | ||||||
|   name = datetime; |   name = datetime; | ||||||
|   cmos = lib/cmos_datetime.c; |   cmos = lib/cmos_datetime.c; | ||||||
|   efi = lib/efi/datetime.c; |   efi = lib/efi/datetime.c; | ||||||
|  |   uboot = lib/uboot/datetime.c; | ||||||
|   sparc64_ieee1275 = lib/ieee1275/datetime.c; |   sparc64_ieee1275 = lib/ieee1275/datetime.c; | ||||||
|   powerpc_ieee1275 = lib/ieee1275/datetime.c; |   powerpc_ieee1275 = lib/ieee1275/datetime.c; | ||||||
|   sparc64_ieee1275 = lib/ieee1275/cmos.c; |   sparc64_ieee1275 = lib/ieee1275/cmos.c; | ||||||
|  | @ -1377,6 +1401,7 @@ module = { | ||||||
|   extra_dist = lib/powerpc/setjmp.S; |   extra_dist = lib/powerpc/setjmp.S; | ||||||
|   extra_dist = lib/ia64/setjmp.S; |   extra_dist = lib/ia64/setjmp.S; | ||||||
|   extra_dist = lib/ia64/longjmp.S; |   extra_dist = lib/ia64/longjmp.S; | ||||||
|  |   extra_dist = lib/arm/setjmp.S; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| module = { | module = { | ||||||
|  | @ -1456,9 +1481,12 @@ module = { | ||||||
|   powerpc_ieee1275 = loader/powerpc/ieee1275/linux.c; |   powerpc_ieee1275 = loader/powerpc/ieee1275/linux.c; | ||||||
|   sparc64_ieee1275 = loader/sparc64/ieee1275/linux.c; |   sparc64_ieee1275 = loader/sparc64/ieee1275/linux.c; | ||||||
|   ia64_efi = loader/ia64/efi/linux.c; |   ia64_efi = loader/ia64/efi/linux.c; | ||||||
|  |   arm = loader/arm/linux.c; | ||||||
|   common = loader/linux.c; |   common = loader/linux.c; | ||||||
|   common = lib/cmdline.c; |   common = lib/cmdline.c; | ||||||
|   enable = noemu; |   enable = noemu; | ||||||
|  | 
 | ||||||
|  |   fdt_cppflags = '$(CPPFLAGS_LIBFDT)'; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| module = { | module = { | ||||||
|  | @ -1507,6 +1535,7 @@ module = { | ||||||
| 
 | 
 | ||||||
|   enable = x86; |   enable = x86; | ||||||
|   enable = ia64_efi; |   enable = ia64_efi; | ||||||
|  |   enable = arm_efi; | ||||||
|   enable = mips; |   enable = mips; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										346
									
								
								grub-core/disk/uboot/ubootdisk.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										346
									
								
								grub-core/disk/uboot/ubootdisk.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,346 @@ | ||||||
|  | /* ubootdisk.c - disk subsystem support for U-Boot platforms */ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/disk.h> | ||||||
|  | #include <grub/err.h> | ||||||
|  | #include <grub/misc.h> | ||||||
|  | #include <grub/mm.h> | ||||||
|  | #include <grub/partition.h> | ||||||
|  | #include <grub/term.h> | ||||||
|  | #include <grub/types.h> | ||||||
|  | #include <grub/uboot/disk.h> | ||||||
|  | #include <grub/uboot/uboot.h> | ||||||
|  | 
 | ||||||
|  | static struct ubootdisk_data *fd_devices; | ||||||
|  | static struct ubootdisk_data *hd_devices; | ||||||
|  | static struct ubootdisk_data *cd_devices; | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * grub_ubootdisk_register(): | ||||||
|  |  *   Called for each disk device enumerated as part of U-Boot initialization | ||||||
|  |  *   code. | ||||||
|  |  */ | ||||||
|  | grub_err_t | ||||||
|  | grub_ubootdisk_register (struct device_info *newdev, int handle) | ||||||
|  | { | ||||||
|  |   struct ubootdisk_data *d; | ||||||
|  |   enum disktype type; | ||||||
|  | 
 | ||||||
|  | #define STOR_TYPE(x) ((x) & 0x0ff0) | ||||||
|  |   switch (STOR_TYPE (newdev->type)) | ||||||
|  |     { | ||||||
|  |     case DT_STOR_IDE: | ||||||
|  |     case DT_STOR_SATA: | ||||||
|  |       /* hd */ | ||||||
|  |       type = hd; | ||||||
|  |       break; | ||||||
|  |     case DT_STOR_MMC: | ||||||
|  |     case DT_STOR_USB: | ||||||
|  |       /* fd */ | ||||||
|  |       type = fd; | ||||||
|  |       break; | ||||||
|  |     default: | ||||||
|  |       return GRUB_ERR_BAD_DEVICE; | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   d = (struct ubootdisk_data *) grub_malloc (sizeof (struct ubootdisk_data)); | ||||||
|  |   if (!d) | ||||||
|  |     return GRUB_ERR_OUT_OF_MEMORY; | ||||||
|  |   d->handle = handle; | ||||||
|  |   d->cookie = newdev->cookie; | ||||||
|  |   d->opencount = 0; | ||||||
|  | 
 | ||||||
|  |   switch (type) | ||||||
|  |     { | ||||||
|  |     case cd: | ||||||
|  |       grub_dprintf ("ubootdisk", "registering cd device\n"); | ||||||
|  |       d->next = cd_devices; | ||||||
|  |       cd_devices = d; | ||||||
|  | 
 | ||||||
|  |       break; | ||||||
|  |     case fd: | ||||||
|  |       grub_dprintf ("ubootdisk", "registering fd device\n"); | ||||||
|  |       d->next = fd_devices; | ||||||
|  |       fd_devices = d; | ||||||
|  | 
 | ||||||
|  |       break; | ||||||
|  |     case hd: | ||||||
|  |       grub_dprintf ("ubootdisk", "registering hd device\n"); | ||||||
|  |       d->next = hd_devices; | ||||||
|  |       hd_devices = d; | ||||||
|  | 
 | ||||||
|  |       break; | ||||||
|  |     default: | ||||||
|  |       grub_free (d); | ||||||
|  |       return GRUB_ERR_BAD_DEVICE; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * uboot_disk_iterate(): | ||||||
|  |  *   Itarator over enumerated disk devices. | ||||||
|  |  */ | ||||||
|  | static int | ||||||
|  | uboot_disk_iterate (grub_disk_dev_iterate_hook_t hook, void *hook_data, | ||||||
|  | 		    grub_disk_pull_t pull) | ||||||
|  | { | ||||||
|  |   struct ubootdisk_data *d; | ||||||
|  |   char buf[16]; | ||||||
|  |   int count; | ||||||
|  | 
 | ||||||
|  |   switch (pull) | ||||||
|  |     { | ||||||
|  |     case GRUB_DISK_PULL_NONE: | ||||||
|  |       /* "hd" - built-in mass-storage */ | ||||||
|  |       for (d = hd_devices, count = 0; d; d = d->next, count++) | ||||||
|  | 	{ | ||||||
|  | 	  grub_snprintf (buf, sizeof (buf) - 1, "hd%d", count); | ||||||
|  | 	  grub_dprintf ("ubootdisk", "iterating %s\n", buf); | ||||||
|  | 	  if (hook (buf, hook_data)) | ||||||
|  | 	    return 1; | ||||||
|  | 	} | ||||||
|  |       break; | ||||||
|  |     case GRUB_DISK_PULL_REMOVABLE: | ||||||
|  |       /* "floppy" - removable mass storage */ | ||||||
|  |       for (d = fd_devices, count = 0; d; d = d->next, count++) | ||||||
|  | 	{ | ||||||
|  | 	  grub_snprintf (buf, sizeof (buf) - 1, "fd%d", count); | ||||||
|  | 	  grub_dprintf ("ubootdisk", "iterating %s\n", buf); | ||||||
|  | 	  if (hook (buf, hook_data)) | ||||||
|  | 	    return 1; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  |       /* "cdrom" - removeable read-only storage */ | ||||||
|  |       for (d = cd_devices, count = 0; d; d = d->next, count++) | ||||||
|  | 	{ | ||||||
|  | 	  grub_snprintf (buf, sizeof (buf) - 1, "cd%d", count); | ||||||
|  | 	  grub_dprintf ("ubootdisk", "iterating %s\n", buf); | ||||||
|  | 	  if (hook (buf, hook_data)) | ||||||
|  | 	    return 1; | ||||||
|  | 	} | ||||||
|  |       break; | ||||||
|  |     default: | ||||||
|  |       return 0; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Helper function for uboot_disk_open. */ | ||||||
|  | static struct ubootdisk_data * | ||||||
|  | get_device (struct ubootdisk_data *devices, int num) | ||||||
|  | { | ||||||
|  |   struct ubootdisk_data *d; | ||||||
|  | 
 | ||||||
|  |   for (d = devices; d && num; d = d->next, num--) | ||||||
|  |     ; | ||||||
|  | 
 | ||||||
|  |   if (num == 0) | ||||||
|  |     return d; | ||||||
|  | 
 | ||||||
|  |   return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * uboot_disk_open(): | ||||||
|  |  *   Opens a disk device already enumerated. | ||||||
|  |  */ | ||||||
|  | static grub_err_t | ||||||
|  | uboot_disk_open (const char *name, struct grub_disk *disk) | ||||||
|  | { | ||||||
|  |   struct ubootdisk_data *d; | ||||||
|  |   struct device_info *devinfo; | ||||||
|  |   int num; | ||||||
|  |   int retval; | ||||||
|  | 
 | ||||||
|  |   grub_dprintf ("ubootdisk", "Opening '%s'\n", name); | ||||||
|  | 
 | ||||||
|  |   num = grub_strtoul (name + 2, 0, 10); | ||||||
|  |   if (grub_errno != GRUB_ERR_NONE) | ||||||
|  |     { | ||||||
|  |       grub_dprintf ("ubootdisk", "Opening '%s' failed, invalid number\n", | ||||||
|  | 		    name); | ||||||
|  |       goto fail; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   if (name[1] != 'd') | ||||||
|  |     { | ||||||
|  |       grub_dprintf ("ubootdisk", "Opening '%s' failed, invalid name\n", name); | ||||||
|  |       goto fail; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   switch (name[0]) | ||||||
|  |     { | ||||||
|  |     case 'f': | ||||||
|  |       d = get_device (fd_devices, num); | ||||||
|  |       break; | ||||||
|  |     case 'c': | ||||||
|  |       d = get_device (cd_devices, num); | ||||||
|  |       break; | ||||||
|  |     case 'h': | ||||||
|  |       d = get_device (hd_devices, num); | ||||||
|  |       break; | ||||||
|  |     default: | ||||||
|  |       goto fail; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   if (!d) | ||||||
|  |     goto fail; | ||||||
|  | 
 | ||||||
|  |   /*
 | ||||||
|  |    * Subsystems may call open on the same device recursively - but U-Boot | ||||||
|  |    * does not deal with this. So simply keep track of number of calls and | ||||||
|  |    * return success if already open. | ||||||
|  |    */ | ||||||
|  |   if (d->opencount > 0) | ||||||
|  |     { | ||||||
|  |       grub_dprintf ("ubootdisk", "(%s) already open\n", disk->name); | ||||||
|  |       d->opencount++; | ||||||
|  |       retval = 0; | ||||||
|  |     } | ||||||
|  |   else | ||||||
|  |     { | ||||||
|  |       retval = uboot_dev_open (d->handle); | ||||||
|  |       if (retval != 0) | ||||||
|  | 	goto fail; | ||||||
|  |       d->opencount = 1; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   grub_dprintf ("ubootdisk", "cookie: 0x%08x\n", (grub_addr_t) d->cookie); | ||||||
|  |   disk->id = (grub_addr_t) d->cookie; | ||||||
|  | 
 | ||||||
|  |   /* Device has previously been enumerated, so this should never fail */ | ||||||
|  |   if ((devinfo = uboot_dev_get (d->handle)) == NULL) | ||||||
|  |     goto fail; | ||||||
|  | 
 | ||||||
|  |   d->block_size = devinfo->di_stor.block_size; | ||||||
|  |   if (d->block_size == 0) | ||||||
|  |     { | ||||||
|  |       grub_printf ("%s: no block size!\n", __FUNCTION__); | ||||||
|  |       return GRUB_ERR_IO; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   for (disk->log_sector_size = 0; | ||||||
|  |        (1U << disk->log_sector_size) < d->block_size; | ||||||
|  |        disk->log_sector_size++); | ||||||
|  | 
 | ||||||
|  |   grub_dprintf ("ubootdisk", "(%s) blocksize=%d, log_sector_size=%d\n", | ||||||
|  | 		disk->name, d->block_size, disk->log_sector_size); | ||||||
|  | 
 | ||||||
|  |   disk->total_sectors = GRUB_DISK_SIZE_UNKNOWN; | ||||||
|  |   disk->data = d; | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | 
 | ||||||
|  | fail: | ||||||
|  |   return grub_error (GRUB_ERR_UNKNOWN_DEVICE, "no such device"); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void | ||||||
|  | uboot_disk_close (struct grub_disk *disk) | ||||||
|  | { | ||||||
|  |   struct ubootdisk_data *d; | ||||||
|  |   int retval; | ||||||
|  | 
 | ||||||
|  |   d = disk->data; | ||||||
|  | 
 | ||||||
|  |   /*
 | ||||||
|  |    * In mirror of open function, keep track of number of calls to close and | ||||||
|  |    * send on to U-Boot only when opencount would decrease to 0. | ||||||
|  |    */ | ||||||
|  |   if (d->opencount > 1) | ||||||
|  |     { | ||||||
|  |       grub_dprintf ("ubootdisk", "Closed (%s)\n", disk->name); | ||||||
|  | 
 | ||||||
|  |       d->opencount--; | ||||||
|  |     } | ||||||
|  |   else if (d->opencount == 1) | ||||||
|  |     { | ||||||
|  |       retval = uboot_dev_close (d->handle); | ||||||
|  |       d->opencount--; | ||||||
|  |       grub_dprintf ("ubootdisk", "closed %s (%d)\n", disk->name, retval); | ||||||
|  |     } | ||||||
|  |   else | ||||||
|  |     { | ||||||
|  |       grub_dprintf ("ubootdisk", "device %s not open!\n", disk->name); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * uboot_disk_read(): | ||||||
|  |  *   Called from within disk subsystem to read a sequence of blocks into the | ||||||
|  |  *   disk cache. Maps directly on top of U-Boot API, only wrap in some error | ||||||
|  |  *   handling. | ||||||
|  |  */ | ||||||
|  | static grub_err_t | ||||||
|  | uboot_disk_read (struct grub_disk *disk, | ||||||
|  | 		 grub_disk_addr_t offset, grub_size_t numblocks, char *buf) | ||||||
|  | { | ||||||
|  |   struct ubootdisk_data *d; | ||||||
|  |   lbasize_t real_size; | ||||||
|  |   int retval; | ||||||
|  | 
 | ||||||
|  |   d = disk->data; | ||||||
|  | 
 | ||||||
|  |   retval = uboot_dev_read (d->handle, buf, numblocks, offset, &real_size); | ||||||
|  |   grub_dprintf ("ubootdisk", | ||||||
|  | 		"retval=%d, numblocks=%d, real_size=%llu, sector=%llu\n", | ||||||
|  | 		retval, numblocks, (grub_uint64_t) real_size, | ||||||
|  | 		(grub_uint64_t) offset); | ||||||
|  |   if (retval != 0) | ||||||
|  |     return grub_error (GRUB_ERR_IO, "U-Boot disk read error"); | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static grub_err_t | ||||||
|  | uboot_disk_write (struct grub_disk *disk __attribute__ ((unused)), | ||||||
|  | 		  grub_disk_addr_t sector __attribute__ ((unused)), | ||||||
|  | 		  grub_size_t size __attribute__ ((unused)), | ||||||
|  | 		  const char *buf __attribute__ ((unused))) | ||||||
|  | { | ||||||
|  |   grub_dprintf ("ubootdisk", "attempt to write\n"); | ||||||
|  |   return GRUB_ERR_NOT_IMPLEMENTED_YET; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static struct grub_disk_dev grub_ubootdisk_dev = { | ||||||
|  |   .name = "ubootdisk", | ||||||
|  |   .id = GRUB_DISK_DEVICE_UBOOTDISK_ID, | ||||||
|  |   .iterate = uboot_disk_iterate, | ||||||
|  |   .open = uboot_disk_open, | ||||||
|  |   .close = uboot_disk_close, | ||||||
|  |   .read = uboot_disk_read, | ||||||
|  |   .write = uboot_disk_write, | ||||||
|  |   .next = 0 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | grub_ubootdisk_init (void) | ||||||
|  | { | ||||||
|  |   grub_disk_dev_register (&grub_ubootdisk_dev); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | grub_ubootdisk_fini (void) | ||||||
|  | { | ||||||
|  |   grub_disk_dev_unregister (&grub_ubootdisk_dev); | ||||||
|  | } | ||||||
							
								
								
									
										251
									
								
								grub-core/kern/arm/cache.S
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										251
									
								
								grub-core/kern/arm/cache.S
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,251 @@ | ||||||
|  | /* | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/symbol.h> | ||||||
|  | #include <grub/dl.h> | ||||||
|  | 
 | ||||||
|  | 	.file	"cache.S" | ||||||
|  | 	.text | ||||||
|  | 	.syntax	unified
 | ||||||
|  | #if !defined (__thumb2__) | ||||||
|  | 	.arm | ||||||
|  | #define ARM(x...)	x | ||||||
|  | #define THUMB(x...) | ||||||
|  | #else | ||||||
|  | 	.thumb | ||||||
|  | #define THUMB(x...)	x | ||||||
|  | #define ARM(x...) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | 	.align	2
 | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  |  * Simple cache maintenance functions | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | @ r0 - *beg (inclusive)
 | ||||||
|  | @ r1 - *end (exclusive)	
 | ||||||
|  | clean_dcache_range: | ||||||
|  | 	@ Clean data cache range for range to point-of-unification
 | ||||||
|  | 	ldr	r2, dlinesz | ||||||
|  | 1:	cmp	r0, r1 | ||||||
|  | 	bge	2f | ||||||
|  | #ifdef DEBUG | ||||||
|  | 	push	{r0-r2, lr} | ||||||
|  | 	mov	r1, r2 | ||||||
|  | 	mov	r2, r0 | ||||||
|  | 	ldr	r0, =dcstr | ||||||
|  | 	bl	EXT_C(grub_printf) | ||||||
|  | 	pop	{r0-r2, lr} | ||||||
|  | #endif | ||||||
|  | 	mcr	p15, 0, r0, c7, c11, 1	@ DCCMVAU
 | ||||||
|  | 	add	r0, r0, r2		@ Next line
 | ||||||
|  | 	b	1b | ||||||
|  | 2:	dsb | ||||||
|  | 	bx	lr | ||||||
|  | 
 | ||||||
|  | @ r0 - *beg (inclusive)
 | ||||||
|  | @ r1 - *end (exclusive)	
 | ||||||
|  | invalidate_icache_range: | ||||||
|  | 	@ Invalidate instruction cache for range to point-of-unification
 | ||||||
|  | 	ldr	r2, ilinesz | ||||||
|  | 1:	cmp	r0, r1 | ||||||
|  | 	bge	2f | ||||||
|  | #ifdef DEBUG | ||||||
|  | 	push	{r0-r2, lr} | ||||||
|  | 	mov	r1, r2 | ||||||
|  | 	mov	r2, r0 | ||||||
|  | 	ldr	r0, =icstr | ||||||
|  | 	bl	EXT_C(grub_printf) | ||||||
|  | 	pop	{r0-r2, lr} | ||||||
|  | #endif | ||||||
|  | 	mcr	p15, 0, r0, c7, c5, 1	@ ICIMVAU
 | ||||||
|  | 	add	r0, r0, r2		@ Next line
 | ||||||
|  | 	b	1b | ||||||
|  | 	@ Branch predictor invalidate all
 | ||||||
|  | 2:	mcr	p15, 0, r0, c7,	c5, 6	@ BPIALL
 | ||||||
|  | 	dsb | ||||||
|  | 	isb | ||||||
|  | 	bx	lr | ||||||
|  | 	 | ||||||
|  | @void __wrap___clear_cache(char *beg, char *end);
 | ||||||
|  | FUNCTION(__wrap___clear_cache) | ||||||
|  | 	dmb | ||||||
|  | 	dsb | ||||||
|  | 	push	{r4-r6, lr} | ||||||
|  | 	ldr	r2, probed	@ If first call, probe cache sizes
 | ||||||
|  | 	cmp	r2, #0 | ||||||
|  | 	bleq	probe_caches	@ This call corrupts r3
 | ||||||
|  | 	mov	r4, r0 | ||||||
|  | 	mov	r5, r1 | ||||||
|  | 	bl	clean_dcache_range | ||||||
|  | 	mov	r0, r4 | ||||||
|  | 	mov	r1, r5 | ||||||
|  | 	bl	invalidate_icache_range | ||||||
|  | 	pop	{r4-r6, pc} | ||||||
|  | 
 | ||||||
|  | probe_caches: | ||||||
|  | 	push	{r4-r6, lr} | ||||||
|  | 	mrc 	p15, 0, r4, c0, c0, 1	@ Read Cache Type Register
 | ||||||
|  | 	mov	r5, #1 | ||||||
|  | 	ubfx	r6, r4, #16, #4		@ Extract min D-cache num word log2
 | ||||||
|  | 	add	r6, r6, #2		@ words->bytes
 | ||||||
|  | 	lsl	r6, r5, r6		@ Convert to num bytes
 | ||||||
|  | 	ldr	r3, =dlinesz | ||||||
|  | 	str	r6, [r3] | ||||||
|  | 	and	r6, r4, #0xf		@ Extract min I-cache num word log2
 | ||||||
|  | 	add	r6, r6, #2		@ words->bytes
 | ||||||
|  | 	lsl	r6, r5, r6		@ Convert to num bytes
 | ||||||
|  | 	ldr	r3, =ilinesz | ||||||
|  | 	str	r6, [r3] | ||||||
|  | 	ldr	r3, =probed		@ Flag cache probing done
 | ||||||
|  | 	str	r5, [r3] | ||||||
|  | 	pop	{r4-r6, pc} | ||||||
|  | 
 | ||||||
|  | #ifdef DEBUG | ||||||
|  | dcstr:	.asciz	"cleaning %d bytes of D cache @ 0x%08x\n" | ||||||
|  | icstr:	.asciz	"invalidating %d bytes of I cache @ 0x%08x\n" | ||||||
|  | #endif | ||||||
|  | 	 | ||||||
|  | 	.align	3
 | ||||||
|  | probed:	.long	0 | ||||||
|  | dlinesz: | ||||||
|  | 	.long	0
 | ||||||
|  | ilinesz: | ||||||
|  | 	.long	0
 | ||||||
|  | 
 | ||||||
|  | @void grub_arch_sync_caches (void *address, grub_size_t len)
 | ||||||
|  | FUNCTION(grub_arch_sync_caches) | ||||||
|  | 	add	r1, r0, r1 | ||||||
|  | 	b	__wrap___clear_cache | ||||||
|  | 
 | ||||||
|  | 	@ r0  - CLIDR
 | ||||||
|  | 	@ r1  - LoC
 | ||||||
|  | 	@ r2  - current level
 | ||||||
|  | 	@ r3  - num sets
 | ||||||
|  | 	@ r4  - num ways
 | ||||||
|  | 	@ r5  - current set
 | ||||||
|  | 	@ r6  - current way
 | ||||||
|  | 	@ r7  - line size
 | ||||||
|  | 	@ r8  - scratch
 | ||||||
|  | 	@ r9  - scratch
 | ||||||
|  | 	@ r10 - scratch
 | ||||||
|  | 	@ r11 - scratch
 | ||||||
|  | clean_invalidate_dcache: | ||||||
|  | 	push	{r4-r12, lr} | ||||||
|  | 	mrc 	p15, 1, r0, c0, c0, 1	@ Read CLIDR
 | ||||||
|  | 	ubfx	r1, r0, #24, #3		@ Extract LoC
 | ||||||
|  | 	 | ||||||
|  | 	mov	r2, #0			@ First level, L1
 | ||||||
|  | 2:	and	r8, r0, #7		@ cache type at current level
 | ||||||
|  | 	cmp	r8, #2 | ||||||
|  | 	blt	5f			@ instruction only, or none, skip level
 | ||||||
|  | 
 | ||||||
|  | 	@ set current cache level/type (for CSSIDR read)
 | ||||||
|  | 	lsl	r8, r2, #1 | ||||||
|  | 	mcr	p15, 2, r8, c0, c0, 0	@ Write CSSELR (level, type: data/uni)
 | ||||||
|  | 
 | ||||||
|  | 	@ read current cache information
 | ||||||
|  | 	mrc	p15, 1, r8, c0, c0, 0	@ Read CSSIDR
 | ||||||
|  | 	ubfx	r3, r8, #13, #14	@ Number of sets -1
 | ||||||
|  | 	ubfx	r4, r8, #3, #9		@ Number of ways -1
 | ||||||
|  | 	and	r7, r8, #7		@ log2(line size in words) - 2
 | ||||||
|  | 	add	r7, r7, #2		@  adjust
 | ||||||
|  | 	mov	r8, #1 | ||||||
|  | 	lsl	r7, r8, r7		@  -> line size in words
 | ||||||
|  | 	lsl	r7, r7, #2		@  -> bytes
 | ||||||
|  | 
 | ||||||
|  | 	@ set loop
 | ||||||
|  | 	mov	r5, #0			@ current set = 0
 | ||||||
|  | 3:	lsl	r8, r2, #1		@ insert level
 | ||||||
|  | 	clz	r9, r7			@ calculate set field offset
 | ||||||
|  | 	mov	r10, #31 | ||||||
|  | 	sub	r9, r10, r9 | ||||||
|  | 	lsl	r10, r5, r9 | ||||||
|  | 	orr	r8, r8, r10		@ insert set field
 | ||||||
|  | 
 | ||||||
|  | 	@ way loop
 | ||||||
|  | 	@ calculate way field offset
 | ||||||
|  | 	mov	r6, #0			@ current way = 0
 | ||||||
|  | 	add	r10, r4, #1 | ||||||
|  | 	clz	r9, r10			@ r9 = way field offset
 | ||||||
|  | 	add	r9, r9, #1 | ||||||
|  | 4:	lsl	r10, r6, r9 | ||||||
|  | 	orr	r11, r8, r10		@ insert way field	
 | ||||||
|  | 	 | ||||||
|  | 	@ clean line by set/way
 | ||||||
|  | 	mcr	p15, 0, r11, c7, c14, 2	@ DCCISW
 | ||||||
|  | 	 | ||||||
|  | 	@ next way
 | ||||||
|  | 	add	r6, r6, #1 | ||||||
|  | 	cmp	r6, r4 | ||||||
|  | 	ble	4b | ||||||
|  | 
 | ||||||
|  | 	@ next set
 | ||||||
|  | 	add	r5, r5, #1 | ||||||
|  | 	cmp	r5, r3 | ||||||
|  | 	ble	3b | ||||||
|  | 	 | ||||||
|  | 	@ next level
 | ||||||
|  | 5:	lsr	r0, r0, #3		@ align next level CLIDR 'type' field
 | ||||||
|  | 	add	r2, r2, #1		@ increment cache level counter
 | ||||||
|  | 	cmp	r2, r1 | ||||||
|  | 	blt	2b			@ outer loop
 | ||||||
|  | 
 | ||||||
|  | 	@ return
 | ||||||
|  | 6:	dsb | ||||||
|  | 	isb | ||||||
|  | 	pop	{r4-r12, pc} | ||||||
|  | 
 | ||||||
|  | FUNCTION(grub_arm_disable_caches_mmu) | ||||||
|  | 	push	{r4, lr} | ||||||
|  | 
 | ||||||
|  | 	@ disable D-cache
 | ||||||
|  | 	mrc	p15, 0, r0, c1, c0, 0 | ||||||
|  | 	bic	r0, r0, #(1 << 2) | ||||||
|  | 	mcr	p15, 0, r0, c1, c0, 0 | ||||||
|  | 	dsb | ||||||
|  | 	isb | ||||||
|  | 
 | ||||||
|  | 	@ clean/invalidate D-cache
 | ||||||
|  | 	bl	clean_invalidate_dcache | ||||||
|  | 
 | ||||||
|  | 	@ disable I-cache
 | ||||||
|  | 	mrc	p15, 0, r0, c1, c0, 0 | ||||||
|  | 	bic	r0, r0, #(1 << 12) | ||||||
|  | 	mcr	p15, 0, r0, c1, c0, 0 | ||||||
|  | 	dsb | ||||||
|  | 	isb | ||||||
|  | 
 | ||||||
|  | 	@ invalidate I-cache (also invalidates branch predictors)
 | ||||||
|  | 	mcr	p15, 0, r0, c7, c5, 0 | ||||||
|  | 	dsb | ||||||
|  | 	isb | ||||||
|  | 	 | ||||||
|  | 	@ clear SCTLR M bit
 | ||||||
|  | 	mrc	p15, 0, r0, c1, c0, 0 | ||||||
|  | 	bic	r0, r0, #(1 << 0) | ||||||
|  | 	mcr	p15, 0, r0, c1, c0, 0 | ||||||
|  | 
 | ||||||
|  | 	mcr	p15, 0, r0, c8, c7, 0	@ invalidate TLB
 | ||||||
|  | 	mcr	p15, 0, r0, c7, c5, 6	@ invalidate branch predictor
 | ||||||
|  | 	dsb | ||||||
|  | 	isb | ||||||
|  | 
 | ||||||
|  | 	pop	{r4, pc} | ||||||
|  | 
 | ||||||
							
								
								
									
										490
									
								
								grub-core/kern/arm/dl.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										490
									
								
								grub-core/kern/arm/dl.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,490 @@ | ||||||
|  | /* dl.c - arch-dependent part of loadable module support */ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/dl.h> | ||||||
|  | #include <grub/elf.h> | ||||||
|  | #include <grub/misc.h> | ||||||
|  | #include <grub/err.h> | ||||||
|  | #include <grub/mm.h> | ||||||
|  | #include <grub/i18n.h> | ||||||
|  | 
 | ||||||
|  | #ifdef GRUB_UTIL | ||||||
|  | # include <grub/util/misc.h> | ||||||
|  | #else | ||||||
|  | # if !defined(__thumb2__) | ||||||
|  | #  error "Relocations not implemented for A32 ("ARM") instruction set yet!" | ||||||
|  | # endif | ||||||
|  | 
 | ||||||
|  | grub_err_t reloc_jump24 (grub_uint32_t *addr, Elf32_Addr sym_addr); | ||||||
|  | grub_err_t reloc_thm_call (grub_uint16_t *addr, Elf32_Addr sym_addr); | ||||||
|  | grub_err_t reloc_thm_jump19 (grub_uint16_t *addr, Elf32_Addr sym_addr); | ||||||
|  | 
 | ||||||
|  | #ifdef DL_DEBUG | ||||||
|  | static const char *symstrtab; | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * This is a bit of a hack, setting the symstrtab pointer to the last STRTAB | ||||||
|  |  * section in the module (which is where symbol names are in the objects I've | ||||||
|  |  * inspected manually).  | ||||||
|  |  */ | ||||||
|  | static void | ||||||
|  | set_symstrtab (Elf_Ehdr * e) | ||||||
|  | { | ||||||
|  |   int i; | ||||||
|  |   Elf_Shdr *s; | ||||||
|  | 
 | ||||||
|  |   symstrtab = NULL; | ||||||
|  | 
 | ||||||
|  |   for (i = 0, s = (Elf_Shdr *) ((grub_uint32_t) e + e->e_shoff); | ||||||
|  |        i < e->e_shnum; | ||||||
|  |        i++, s = (Elf_Shdr *) ((grub_uint32_t) s + e->e_shentsize)) | ||||||
|  |     if (s->sh_type == SHT_STRTAB) | ||||||
|  |       symstrtab = (void *) ((grub_addr_t) e + s->sh_offset); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static const char * | ||||||
|  | get_symbolname (Elf_Sym * sym) | ||||||
|  | { | ||||||
|  |   const char *symbolname = symstrtab + sym->st_name; | ||||||
|  | 
 | ||||||
|  |   return (*symbolname ? symbolname : NULL); | ||||||
|  | } | ||||||
|  | #endif /* DL_DEBUG */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * R_ARM_ABS32 | ||||||
|  |  * | ||||||
|  |  * Simple relocation of 32-bit value (in literal pool) | ||||||
|  |  */ | ||||||
|  | static grub_err_t | ||||||
|  | reloc_abs32 (Elf_Word *target, Elf_Addr sym_addr) | ||||||
|  | { | ||||||
|  |   Elf_Addr tmp; | ||||||
|  | 
 | ||||||
|  |   tmp = *target; | ||||||
|  |   tmp += sym_addr; | ||||||
|  |   *target = tmp; | ||||||
|  | #if 0 //def GRUB_UTIL
 | ||||||
|  |   grub_util_info ("  %s:  reloc_abs32 0x%08x => 0x%08x", __FUNCTION__, | ||||||
|  | 		  (unsigned int) sym_addr, (unsigned int) tmp); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | } | ||||||
|  | #endif /* ndef GRUB_UTIL */ | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /********************************************************************
 | ||||||
|  |  * Thumb (T32) relocations:                                         * | ||||||
|  |  *                                                                  * | ||||||
|  |  * 32-bit Thumb instructions can be 16-bit aligned, and are fetched * | ||||||
|  |  * little-endian, requiring some additional fiddling.               * | ||||||
|  |  ********************************************************************/ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * R_ARM_THM_CALL/THM_JUMP24 | ||||||
|  |  * | ||||||
|  |  * Relocate Thumb (T32) instruction set relative branches: | ||||||
|  |  *   B.W, BL and BLX | ||||||
|  |  */ | ||||||
|  | grub_err_t | ||||||
|  | reloc_thm_call (grub_uint16_t *target, Elf32_Addr sym_addr) | ||||||
|  | { | ||||||
|  |   grub_int32_t offset, offset_low, offset_high; | ||||||
|  |   grub_uint32_t sign, j1, j2, is_blx; | ||||||
|  |   grub_uint32_t insword, insmask; | ||||||
|  | 
 | ||||||
|  |   /* Extract instruction word in alignment-safe manner */ | ||||||
|  |   insword = (*target << 16) | (*(target + 1)); | ||||||
|  |   insmask = 0xf800d000; | ||||||
|  | 
 | ||||||
|  |   /* B.W/BL or BLX? Affects range and expected target state */ | ||||||
|  |   if (((insword >> 12) & 0xd) == 0xc) | ||||||
|  |     is_blx = 1; | ||||||
|  |   else | ||||||
|  |     is_blx = 0; | ||||||
|  | 
 | ||||||
|  |   /* If BLX, target symbol must be ARM (target address LSB == 0) */ | ||||||
|  |   if (is_blx && (sym_addr & 1)) | ||||||
|  |     { | ||||||
|  | #ifndef GRUB_UTIL | ||||||
|  |       return grub_error | ||||||
|  | 	(GRUB_ERR_BUG, N_("Relocation targeting wrong execution state")); | ||||||
|  | #else | ||||||
|  |       grub_util_error ("Relocation targeting wrong execution state"); | ||||||
|  | #endif | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   offset_low = -16777216; | ||||||
|  |   offset_high = is_blx ? 16777212 : 16777214; | ||||||
|  | 
 | ||||||
|  |   /* Extract bitfields from instruction words */ | ||||||
|  |   sign = (insword >> 26) & 1; | ||||||
|  |   j1 = (insword >> 13) & 1; | ||||||
|  |   j2 = (insword >> 11) & 1; | ||||||
|  |   offset = (sign << 24) | ((~(j1 ^ sign) & 1) << 23) | | ||||||
|  |     ((~(j2 ^ sign) & 1) << 22) | | ||||||
|  |     ((insword & 0x03ff0000) >> 4) | ((insword & 0x000007ff) << 1); | ||||||
|  | 
 | ||||||
|  |   /* Sign adjust and calculate offset */ | ||||||
|  |   if (offset & (1 << 24)) | ||||||
|  |     offset -= (1 << 25); | ||||||
|  | #ifdef GRUB_UTIL | ||||||
|  |   grub_util_info ("    sym_addr = 0x%08x", sym_addr); | ||||||
|  | #endif | ||||||
|  | #ifdef GRUB_UTIL | ||||||
|  |   offset += sym_addr; | ||||||
|  | #else | ||||||
|  |   offset += sym_addr - (grub_uint32_t) target; | ||||||
|  | #endif | ||||||
|  | #ifdef DEBUG | ||||||
|  |   grub_printf(" %s: target=0x%08x, sym_addr=0x%08x, offset=%d\n", | ||||||
|  | 	      is_blx ? "BLX" : "BL", (unsigned int) target, sym_addr, offset); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |   if ((offset < offset_low) || (offset > offset_high)) | ||||||
|  |     { | ||||||
|  | #ifdef GRUB_UTIL | ||||||
|  |       grub_util_error ("Relocation out of range"); | ||||||
|  | #else | ||||||
|  |       return grub_error | ||||||
|  | 	(GRUB_ERR_OUT_OF_RANGE, N_("THM_CALL Relocation out of range.")); | ||||||
|  | #endif | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | #ifdef GRUB_UTIL | ||||||
|  |   grub_util_info ("    relative destination = 0x%08x", | ||||||
|  | 		  (unsigned int)target + offset); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |   /* Reassemble instruction word */ | ||||||
|  |   sign = (offset >> 24) & 1; | ||||||
|  |   j1 = sign ^ (~(offset >> 23) & 1); | ||||||
|  |   j2 = sign ^ (~(offset >> 22) & 1); | ||||||
|  |   insword = (insword & insmask) | | ||||||
|  |     (sign << 26) | | ||||||
|  |     (((offset >> 12) & 0x03ff) << 16) | | ||||||
|  |     (j1 << 13) | (j2 << 11) | ((offset >> 1) & 0x07ff); | ||||||
|  | 
 | ||||||
|  |   /* Write instruction word back in alignment-safe manner */ | ||||||
|  |   *target = (insword >> 16) & 0xffff; | ||||||
|  |   *(target + 1) = insword & 0xffff; | ||||||
|  | 
 | ||||||
|  | #ifdef GRUB_UTIL | ||||||
|  | #pragma GCC diagnostic ignored "-Wcast-align" | ||||||
|  |   grub_util_info ("    *target = 0x%08x", *((unsigned int *)target)); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * R_ARM_THM_JUMP19 | ||||||
|  |  * | ||||||
|  |  * Relocate conditional Thumb (T32) B<c>.W | ||||||
|  |  */ | ||||||
|  | grub_err_t | ||||||
|  | reloc_thm_jump19 (grub_uint16_t *addr, Elf32_Addr sym_addr) | ||||||
|  | { | ||||||
|  |   grub_int32_t offset; | ||||||
|  |   grub_uint32_t insword, insmask; | ||||||
|  | 
 | ||||||
|  |   /* Extract instruction word in alignment-safe manner */ | ||||||
|  |   insword = (*addr) << 16 | *(addr + 1); | ||||||
|  |   insmask = 0xfbc0d800; | ||||||
|  | 
 | ||||||
|  |   /* Extract and sign extend offset */ | ||||||
|  |   offset = ((insword >> 26) & 1) << 18 | ||||||
|  |     | ((insword >> 11) & 1) << 17 | ||||||
|  |     | ((insword >> 13) & 1) << 16 | ||||||
|  |     | ((insword >> 16) & 0x3f) << 11 | ||||||
|  |     | (insword & 0x7ff); | ||||||
|  |   offset <<= 1; | ||||||
|  |   if (offset & (1 << 19)) | ||||||
|  |     offset -= (1 << 20); | ||||||
|  | 
 | ||||||
|  |   /* Adjust and re-truncate offset */ | ||||||
|  | #ifdef GRUB_UTIL | ||||||
|  |   offset += sym_addr; | ||||||
|  | #else | ||||||
|  |   offset += sym_addr - (grub_uint32_t) addr; | ||||||
|  | #endif | ||||||
|  |   if ((offset > 1048574) || (offset < -1048576)) | ||||||
|  |     { | ||||||
|  |       return grub_error | ||||||
|  |         (GRUB_ERR_OUT_OF_RANGE, N_("THM_JUMP19 Relocation out of range.")); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   offset >>= 1; | ||||||
|  |   offset &= 0x7ffff; | ||||||
|  | 
 | ||||||
|  |   /* Reassemble instruction word and write back */ | ||||||
|  |   insword &= insmask; | ||||||
|  |   insword |= ((offset >> 18) & 1) << 26 | ||||||
|  |     | ((offset >> 17) & 1) << 11 | ||||||
|  |     | ((offset >> 16) & 1) << 13 | ||||||
|  |     | ((offset >> 11) & 0x3f) << 16 | ||||||
|  |     | (offset & 0x7ff); | ||||||
|  |   *addr = insword >> 16; | ||||||
|  |   *(addr + 1) = insword & 0xffff; | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /***********************************************************
 | ||||||
|  |  * ARM (A32) relocations:                                  * | ||||||
|  |  *                                                         * | ||||||
|  |  * ARM instructions are 32-bit in size and 32-bit aligned. * | ||||||
|  |  ***********************************************************/ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * R_ARM_JUMP24 | ||||||
|  |  * | ||||||
|  |  * Relocate ARM (A32) B | ||||||
|  |  */ | ||||||
|  | grub_err_t | ||||||
|  | reloc_jump24 (grub_uint32_t *addr, Elf32_Addr sym_addr) | ||||||
|  | { | ||||||
|  |   grub_uint32_t insword; | ||||||
|  |   grub_int32_t offset; | ||||||
|  | 
 | ||||||
|  |   insword = *addr; | ||||||
|  | 
 | ||||||
|  |   offset = (insword & 0x00ffffff) << 2; | ||||||
|  |   if (offset & 0x02000000) | ||||||
|  |     offset -= 0x04000000; | ||||||
|  | #ifdef GRUB_UTIL | ||||||
|  |   offset += sym_addr; | ||||||
|  | #else | ||||||
|  |   offset += sym_addr - (grub_uint32_t) addr; | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |   insword &= 0xff000000; | ||||||
|  |   insword |= (offset >> 2) & 0x00ffffff; | ||||||
|  | 
 | ||||||
|  |   *addr = insword; | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /*************************************************
 | ||||||
|  |  * Runtime dynamic linker with helper functions. * | ||||||
|  |  *************************************************/ | ||||||
|  | #ifndef GRUB_UTIL | ||||||
|  | /*
 | ||||||
|  |  * find_segment(): finds a module segment matching sh_info | ||||||
|  |  */ | ||||||
|  | static grub_dl_segment_t | ||||||
|  | find_segment (grub_dl_segment_t seg, Elf32_Word sh_info) | ||||||
|  | { | ||||||
|  |   for (; seg; seg = seg->next) | ||||||
|  |     if (seg->section == sh_info) | ||||||
|  |       return seg; | ||||||
|  | 
 | ||||||
|  |   return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * do_relocations(): | ||||||
|  |  *   Iterate over all relocations in section, calling appropriate functions | ||||||
|  |  *   for patching. | ||||||
|  |  */ | ||||||
|  | static grub_err_t | ||||||
|  | do_relocations (Elf_Shdr * relhdr, Elf_Ehdr * e, grub_dl_t mod) | ||||||
|  | { | ||||||
|  |   grub_dl_segment_t seg; | ||||||
|  |   Elf_Rel *rel; | ||||||
|  |   Elf_Sym *sym; | ||||||
|  |   int i, entnum; | ||||||
|  | 
 | ||||||
|  |   entnum = relhdr->sh_size / sizeof (Elf_Rel); | ||||||
|  | 
 | ||||||
|  |   /* Find the target segment for this relocation section. */ | ||||||
|  |   seg = find_segment (mod->segment, relhdr->sh_info); | ||||||
|  |   if (!seg) | ||||||
|  |     return grub_error (GRUB_ERR_EOF, N_("relocation segment not found")); | ||||||
|  | 
 | ||||||
|  |   rel = (Elf_Rel *) ((grub_addr_t) e + relhdr->sh_offset); | ||||||
|  | 
 | ||||||
|  |   /* Step through all relocations */ | ||||||
|  |   for (i = 0, sym = mod->symtab; i < entnum; i++) | ||||||
|  |     { | ||||||
|  |       Elf_Addr *target, sym_addr; | ||||||
|  |       int relsym, reltype; | ||||||
|  |       grub_err_t retval; | ||||||
|  | 
 | ||||||
|  |       if (seg->size < rel[i].r_offset) | ||||||
|  | 	return grub_error (GRUB_ERR_BAD_MODULE, | ||||||
|  | 			   "reloc offset is out of the segment"); | ||||||
|  |       relsym = ELF_R_SYM (rel[i].r_info); | ||||||
|  |       reltype = ELF_R_TYPE (rel[i].r_info); | ||||||
|  |       target = (Elf_Word *) ((grub_addr_t) seg->addr + rel[i].r_offset); | ||||||
|  | 
 | ||||||
|  |       sym_addr = sym[relsym].st_value; | ||||||
|  | 
 | ||||||
|  | #ifdef DL_DEBUG | ||||||
|  | 
 | ||||||
|  |       grub_printf ("%s: 0x%08x -> %s @ 0x%08x\n", __FUNCTION__, | ||||||
|  | 		   (grub_addr_t) sym_addr, get_symbolname (sym), sym->st_value); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |       switch (reltype) | ||||||
|  | 	{ | ||||||
|  | 	case R_ARM_ABS32: | ||||||
|  | 	  { | ||||||
|  | 	    /* Data will be naturally aligned */ | ||||||
|  | 	    retval = reloc_abs32 (target, sym_addr); | ||||||
|  | 	    if (retval != GRUB_ERR_NONE) | ||||||
|  | 	      return retval; | ||||||
|  | 	  } | ||||||
|  | 	  break; | ||||||
|  | 	case R_ARM_JUMP24: | ||||||
|  | 	  { | ||||||
|  | 	    retval = reloc_jump24 (target, sym_addr); | ||||||
|  | 	    if (retval != GRUB_ERR_NONE) | ||||||
|  | 	      return retval; | ||||||
|  | 	  } | ||||||
|  | 	  break; | ||||||
|  | 	case R_ARM_THM_CALL: | ||||||
|  | 	case R_ARM_THM_JUMP24: | ||||||
|  | 	  { | ||||||
|  | 	    /* Thumb instructions can be 16-bit aligned */ | ||||||
|  | 	    retval = reloc_thm_call ((grub_uint16_t *) target, sym_addr); | ||||||
|  | 	    if (retval != GRUB_ERR_NONE) | ||||||
|  | 	      return retval; | ||||||
|  | 	  } | ||||||
|  | 	  break; | ||||||
|  | 	case R_ARM_THM_JUMP19: | ||||||
|  | 	  { | ||||||
|  | 	    /* Thumb instructions can be 16-bit aligned */ | ||||||
|  | 	    retval = reloc_thm_jump19 ((grub_uint16_t *) target, sym_addr); | ||||||
|  | 	    if (retval != GRUB_ERR_NONE) | ||||||
|  | 	      return retval; | ||||||
|  | 	  } | ||||||
|  | 	  break; | ||||||
|  | 	default: | ||||||
|  | 	  return grub_error (GRUB_ERR_NOT_IMPLEMENTED_YET, | ||||||
|  | 			     N_("relocation 0x%x is not implemented yet"), | ||||||
|  | 			     reltype); | ||||||
|  | 	} | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Check if EHDR is a valid ELF header. | ||||||
|  |  */ | ||||||
|  | grub_err_t | ||||||
|  | grub_arch_dl_check_header (void *ehdr) | ||||||
|  | { | ||||||
|  |   Elf_Ehdr *e = ehdr; | ||||||
|  | 
 | ||||||
|  |   /* Check the magic numbers.  */ | ||||||
|  |   if (e->e_ident[EI_CLASS] != ELFCLASS32 | ||||||
|  |       || e->e_ident[EI_DATA] != ELFDATA2LSB || e->e_machine != EM_ARM) | ||||||
|  |     return grub_error (GRUB_ERR_BAD_OS, | ||||||
|  | 		       N_("invalid arch-dependent ELF magic")); | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Verify that provided ELF header contains reference to a symbol table | ||||||
|  |  */ | ||||||
|  | static int | ||||||
|  | has_symtab (Elf_Ehdr * e) | ||||||
|  | { | ||||||
|  |   int i; | ||||||
|  |   Elf_Shdr *s; | ||||||
|  | 
 | ||||||
|  |   for (i = 0, s = (Elf_Shdr *) ((grub_uint32_t) e + e->e_shoff); | ||||||
|  |        i < e->e_shnum; | ||||||
|  |        i++, s = (Elf_Shdr *) ((grub_uint32_t) s + e->e_shentsize)) | ||||||
|  |     if (s->sh_type == SHT_SYMTAB) | ||||||
|  |       return 1; | ||||||
|  | 
 | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * grub_arch_dl_relocate_symbols(): | ||||||
|  |  *   Only externally visible function in this file. | ||||||
|  |  *   Locates the relocations section of the ELF object, and calls | ||||||
|  |  *   do_relocations() to deal with it. | ||||||
|  |  */ | ||||||
|  | grub_err_t | ||||||
|  | grub_arch_dl_relocate_symbols (grub_dl_t mod, void *ehdr) | ||||||
|  | { | ||||||
|  |   Elf_Ehdr *e = ehdr; | ||||||
|  |   Elf_Shdr *s; | ||||||
|  |   unsigned i; | ||||||
|  | 
 | ||||||
|  |   if (!has_symtab (e)) | ||||||
|  |     return grub_error (GRUB_ERR_BAD_MODULE, N_("no symbol table")); | ||||||
|  | 
 | ||||||
|  | #ifdef DL_DEBUG | ||||||
|  |   set_symstrtab (e); | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #define FIRST_SHDR(x) ((Elf_Shdr *) ((grub_addr_t)(x) + (x)->e_shoff)) | ||||||
|  | #define NEXT_SHDR(x, y) ((Elf_Shdr *) ((grub_addr_t)(y) + (x)->e_shentsize)) | ||||||
|  | 
 | ||||||
|  |   for (i = 0, s = FIRST_SHDR (e); i < e->e_shnum; i++, s = NEXT_SHDR (e, s)) | ||||||
|  |     { | ||||||
|  |       grub_err_t ret; | ||||||
|  | 
 | ||||||
|  |       switch (s->sh_type) | ||||||
|  | 	{ | ||||||
|  | 	case SHT_REL: | ||||||
|  | 	  { | ||||||
|  | 	    /* Relocations, no addends */ | ||||||
|  | 	    ret = do_relocations (s, e, mod); | ||||||
|  | 	    if (ret != GRUB_ERR_NONE) | ||||||
|  | 	      return ret; | ||||||
|  | 	  } | ||||||
|  | 	  break; | ||||||
|  | 	case SHT_NULL: | ||||||
|  | 	case SHT_PROGBITS: | ||||||
|  | 	case SHT_SYMTAB: | ||||||
|  | 	case SHT_STRTAB: | ||||||
|  | 	case SHT_NOBITS: | ||||||
|  | 	case SHT_ARM_ATTRIBUTES: | ||||||
|  | 	  break; | ||||||
|  | 	case SHT_RELA: | ||||||
|  | 	default: | ||||||
|  | 	  { | ||||||
|  | 	    grub_printf ("unhandled section_type: %d (0x%08x)\n", | ||||||
|  | 			 s->sh_type, s->sh_type); | ||||||
|  | 	    return GRUB_ERR_NOT_IMPLEMENTED_YET; | ||||||
|  | 	  }; | ||||||
|  | 	} | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | #undef FIRST_SHDR | ||||||
|  | #undef NEXT_SHDR | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | } | ||||||
|  | #endif /* ndef GRUB_UTIL */ | ||||||
							
								
								
									
										68
									
								
								grub-core/kern/arm/efi/init.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								grub-core/kern/arm/efi/init.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,68 @@ | ||||||
|  | /* init.c - initialize an arm-based EFI system */ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013 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/env.h> | ||||||
|  | #include <grub/kernel.h> | ||||||
|  | #include <grub/misc.h> | ||||||
|  | #include <grub/mm.h> | ||||||
|  | #include <grub/time.h> | ||||||
|  | #include <grub/efi/efi.h> | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * A bit ugly, but functional - and should be completely portable. | ||||||
|  |  */ | ||||||
|  | static grub_uint64_t | ||||||
|  | grub_efi_get_time_ms(void) | ||||||
|  | { | ||||||
|  |   grub_efi_time_t now; | ||||||
|  |   grub_uint64_t retval; | ||||||
|  |   grub_efi_status_t status; | ||||||
|  | 
 | ||||||
|  |   status = efi_call_2 (grub_efi_system_table->runtime_services->get_time, | ||||||
|  | 		       &now, NULL); | ||||||
|  |   if (status != GRUB_EFI_SUCCESS) | ||||||
|  |     { | ||||||
|  |       grub_printf("No time!\n"); | ||||||
|  |       return 0; | ||||||
|  |     } | ||||||
|  |   retval = now.year * 365 * 24 * 60 * 60 * 1000; | ||||||
|  |   retval += now.month * 30 * 24 * 60 * 60 * 1000; | ||||||
|  |   retval += now.day * 24 * 60 * 60 * 1000; | ||||||
|  |   retval += now.hour * 60 * 60 * 1000; | ||||||
|  |   retval += now.minute * 60 * 1000; | ||||||
|  |   retval += now.second * 1000; | ||||||
|  |   retval += now.nanosecond / 1000; | ||||||
|  |   | ||||||
|  |   grub_dprintf("timer", "timestamp: 0x%llx\n", retval); | ||||||
|  | 
 | ||||||
|  |   return retval; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | grub_machine_init (void) | ||||||
|  | { | ||||||
|  |   grub_efi_init (); | ||||||
|  |   grub_install_get_time_ms (grub_efi_get_time_ms); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | grub_machine_fini (void) | ||||||
|  | { | ||||||
|  |   grub_efi_fini (); | ||||||
|  | } | ||||||
							
								
								
									
										203
									
								
								grub-core/kern/arm/efi/misc.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										203
									
								
								grub-core/kern/arm/efi/misc.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,203 @@ | ||||||
|  | /* misc.c - various system functions for an arm-based EFI system */ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013 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/misc.h> | ||||||
|  | #include <grub/mm.h> | ||||||
|  | #include <grub/cpu/linux.h> | ||||||
|  | #include <grub/cpu/system.h> | ||||||
|  | #include <grub/efi/efi.h> | ||||||
|  | #include <grub/machine/loader.h> | ||||||
|  | 
 | ||||||
|  | static inline grub_size_t | ||||||
|  | page_align (grub_size_t size) | ||||||
|  | { | ||||||
|  |   return (size + (1 << 12) - 1) & (~((1 << 12) - 1)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Find the optimal number of pages for the memory map. Is it better to
 | ||||||
|  |    move this code to efi/mm.c?  */ | ||||||
|  | static grub_efi_uintn_t | ||||||
|  | find_mmap_size (void) | ||||||
|  | { | ||||||
|  |   static grub_efi_uintn_t mmap_size = 0; | ||||||
|  | 
 | ||||||
|  |   if (mmap_size != 0) | ||||||
|  |     return mmap_size; | ||||||
|  |    | ||||||
|  |   mmap_size = (1 << 12); | ||||||
|  |   while (1) | ||||||
|  |     { | ||||||
|  |       int ret; | ||||||
|  |       grub_efi_memory_descriptor_t *mmap; | ||||||
|  |       grub_efi_uintn_t desc_size; | ||||||
|  |        | ||||||
|  |       mmap = grub_malloc (mmap_size); | ||||||
|  |       if (! mmap) | ||||||
|  | 	return 0; | ||||||
|  | 
 | ||||||
|  |       ret = grub_efi_get_memory_map (&mmap_size, mmap, 0, &desc_size, 0); | ||||||
|  |       grub_free (mmap); | ||||||
|  |        | ||||||
|  |       if (ret < 0) | ||||||
|  | 	{ | ||||||
|  | 	  grub_error (GRUB_ERR_IO, "cannot get memory map"); | ||||||
|  | 	  return 0; | ||||||
|  | 	} | ||||||
|  |       else if (ret > 0) | ||||||
|  | 	break; | ||||||
|  | 
 | ||||||
|  |       mmap_size += (1 << 12); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   /* Increase the size a bit for safety, because GRUB allocates more on
 | ||||||
|  |      later, and EFI itself may allocate more.  */ | ||||||
|  |   mmap_size += (1 << 12); | ||||||
|  | 
 | ||||||
|  |   return page_align (mmap_size); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #define NEXT_MEMORY_DESCRIPTOR(desc, size)      \ | ||||||
|  |   ((grub_efi_memory_descriptor_t *) ((char *) (desc) + (size))) | ||||||
|  | #define PAGE_SHIFT 12 | ||||||
|  | 
 | ||||||
|  | void * | ||||||
|  | grub_efi_allocate_loader_memory (grub_uint32_t min_offset, grub_uint32_t size) | ||||||
|  | { | ||||||
|  |   grub_efi_uintn_t desc_size; | ||||||
|  |   grub_efi_memory_descriptor_t *mmap, *mmap_end; | ||||||
|  |   grub_efi_uintn_t mmap_size, tmp_mmap_size; | ||||||
|  |   grub_efi_memory_descriptor_t *desc; | ||||||
|  |   void *mem = NULL; | ||||||
|  |   grub_addr_t min_start = 0; | ||||||
|  | 
 | ||||||
|  |   mmap_size = find_mmap_size(); | ||||||
|  |   if (!mmap_size) | ||||||
|  |     return NULL; | ||||||
|  | 
 | ||||||
|  |   mmap = grub_malloc(mmap_size); | ||||||
|  |   if (!mmap) | ||||||
|  |     return NULL; | ||||||
|  | 
 | ||||||
|  |   tmp_mmap_size = mmap_size; | ||||||
|  |   if (grub_efi_get_memory_map (&tmp_mmap_size, mmap, 0, &desc_size, 0) <= 0) | ||||||
|  |     { | ||||||
|  |       grub_error (GRUB_ERR_IO, "cannot get memory map"); | ||||||
|  |       goto fail; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   mmap_end = NEXT_MEMORY_DESCRIPTOR (mmap, tmp_mmap_size); | ||||||
|  |   /* Find lowest accessible RAM location */ | ||||||
|  |   { | ||||||
|  |     int found = 0; | ||||||
|  |     for (desc = mmap ; !found && (desc < mmap_end) ; | ||||||
|  | 	 desc = NEXT_MEMORY_DESCRIPTOR(desc, desc_size)) | ||||||
|  |       { | ||||||
|  | 	switch (desc->type) | ||||||
|  | 	  { | ||||||
|  | 	  case GRUB_EFI_CONVENTIONAL_MEMORY: | ||||||
|  | 	  case GRUB_EFI_LOADER_CODE: | ||||||
|  | 	  case GRUB_EFI_LOADER_DATA: | ||||||
|  | 	    min_start = desc->physical_start + min_offset; | ||||||
|  | 	    found = 1; | ||||||
|  | 	    break; | ||||||
|  | 	  default: | ||||||
|  | 	    break; | ||||||
|  | 	  } | ||||||
|  |       } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   /* First, find free pages for the real mode code
 | ||||||
|  |      and the memory map buffer.  */ | ||||||
|  |   for (desc = mmap ; desc < mmap_end ; | ||||||
|  |        desc = NEXT_MEMORY_DESCRIPTOR(desc, desc_size)) | ||||||
|  |     { | ||||||
|  |       grub_uint64_t start, end; | ||||||
|  | 
 | ||||||
|  |       grub_dprintf("mm", "%s: 0x%08x bytes @ 0x%08x\n", | ||||||
|  | 		   __FUNCTION__, | ||||||
|  | 		   (grub_uint32_t) (desc->num_pages << PAGE_SHIFT), | ||||||
|  | 		   (grub_uint32_t) (desc->physical_start)); | ||||||
|  | 
 | ||||||
|  |       if (desc->type != GRUB_EFI_CONVENTIONAL_MEMORY) | ||||||
|  | 	continue; | ||||||
|  | 
 | ||||||
|  |       start = desc->physical_start; | ||||||
|  |       end = start + (desc->num_pages << PAGE_SHIFT); | ||||||
|  |       grub_dprintf("mm", "%s: start=0x%016llx, end=0x%016llx\n", | ||||||
|  | 		  __FUNCTION__, start, end); | ||||||
|  |       start = start < min_start ? min_start : start; | ||||||
|  |       if (start + size > end) | ||||||
|  | 	continue; | ||||||
|  |       grub_dprintf("mm", "%s: let's allocate some (0x%x) pages @ 0x%08x...\n", | ||||||
|  | 		  __FUNCTION__, (size >> PAGE_SHIFT), (grub_addr_t) start); | ||||||
|  |       mem = grub_efi_allocate_pages (start, (size >> PAGE_SHIFT) + 1); | ||||||
|  |       grub_dprintf("mm", "%s: retval=0x%08x\n", | ||||||
|  | 		   __FUNCTION__, (grub_addr_t) mem); | ||||||
|  |       if (! mem) | ||||||
|  | 	{ | ||||||
|  | 	  grub_error (GRUB_ERR_OUT_OF_MEMORY, "cannot allocate memory"); | ||||||
|  | 	  goto fail; | ||||||
|  | 	} | ||||||
|  |       break; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   if (! mem) | ||||||
|  |     { | ||||||
|  |       grub_error (GRUB_ERR_OUT_OF_MEMORY, "cannot allocate memory"); | ||||||
|  |       goto fail; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   grub_free (mmap); | ||||||
|  |   return mem; | ||||||
|  | 
 | ||||||
|  |  fail: | ||||||
|  |   grub_free (mmap); | ||||||
|  |   return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | grub_err_t | ||||||
|  | grub_efi_prepare_platform (void) | ||||||
|  | { | ||||||
|  |   grub_efi_uintn_t mmap_size; | ||||||
|  |   grub_efi_uintn_t map_key; | ||||||
|  |   grub_efi_uintn_t desc_size; | ||||||
|  |   grub_efi_uint32_t desc_version; | ||||||
|  |   grub_efi_memory_descriptor_t *mmap_buf; | ||||||
|  |   grub_err_t err; | ||||||
|  | 
 | ||||||
|  |   /*
 | ||||||
|  |    * Cloned from IA64 | ||||||
|  |    * Must be done after grub_machine_fini because map_key is used by | ||||||
|  |    *exit_boot_services. | ||||||
|  |    */ | ||||||
|  |   mmap_size = find_mmap_size (); | ||||||
|  |   if (! mmap_size) | ||||||
|  |     return GRUB_ERR_OUT_OF_MEMORY; | ||||||
|  |   mmap_buf = grub_efi_allocate_pages (0, page_align (mmap_size) >> 12); | ||||||
|  |   if (! mmap_buf) | ||||||
|  |     return GRUB_ERR_OUT_OF_MEMORY; | ||||||
|  | 
 | ||||||
|  |   err = grub_efi_finish_boot_services (&mmap_size, mmap_buf, &map_key, | ||||||
|  | 				       &desc_size, &desc_version); | ||||||
|  |   if (err != GRUB_ERR_NONE) | ||||||
|  |     return err; | ||||||
|  | 
 | ||||||
|  |   grub_arm_disable_caches_mmu(); | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | } | ||||||
							
								
								
									
										38
									
								
								grub-core/kern/arm/efi/startup.S
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								grub-core/kern/arm/efi/startup.S
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,38 @@ | ||||||
|  | /* | ||||||
|  |  * (C) Copyright 2013 Free Software Foundation | ||||||
|  |  * | ||||||
|  |  * This program 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 2 of
 | ||||||
|  |  * the License, or (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This program 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 this program; if not, write to the Free Software
 | ||||||
|  |  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | ||||||
|  |  * MA 02111-1307 USA | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <grub/symbol.h> | ||||||
|  | 
 | ||||||
|  | 	.file 	"startup.S" | ||||||
|  | 	.text | ||||||
|  | 	.arm | ||||||
|  | FUNCTION(_start) | ||||||
|  | 	/* | ||||||
|  | 	 *  EFI_SYSTEM_TABLE and EFI_HANDLE are passed in r1/r0. | ||||||
|  | 	 */ | ||||||
|  | 	ldr	ip, =EXT_C(grub_efi_image_handle) | ||||||
|  | 	str	r0, [ip] | ||||||
|  | 	ldr	ip, =EXT_C(grub_efi_system_table) | ||||||
|  | 	str	r1, [ip] | ||||||
|  | 	ldr	ip, =EXT_C(grub_main) | ||||||
|  | 	bx	ip | ||||||
|  | 	.thumb		@ For relocation debugging
 | ||||||
|  | 	blx	_start | ||||||
|  | 	.end | ||||||
							
								
								
									
										44
									
								
								grub-core/kern/arm/misc.S
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								grub-core/kern/arm/misc.S
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,44 @@ | ||||||
|  | /* | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/symbol.h> | ||||||
|  | #include <grub/dl.h> | ||||||
|  | 
 | ||||||
|  | 	.file	"misc.S" | ||||||
|  | 	.text | ||||||
|  | 	.syntax	unified
 | ||||||
|  | #if !defined (__thumb2__) | ||||||
|  | 	.arm | ||||||
|  | #define ARM(x...)	x | ||||||
|  | #define THUMB(x...) | ||||||
|  | #else | ||||||
|  | 	.thumb | ||||||
|  | #define THUMB(x...)	x | ||||||
|  | #define ARM(x...) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | 	.align	2
 | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  |  * Null divide-by-zero handler | ||||||
|  |  */ | ||||||
|  | FUNCTION(raise) | ||||||
|  | 	mov	r0, #0 | ||||||
|  | 	bx	lr | ||||||
|  | 	 | ||||||
|  | 	.end | ||||||
							
								
								
									
										176
									
								
								grub-core/kern/arm/uboot/startup.S
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										176
									
								
								grub-core/kern/arm/uboot/startup.S
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,176 @@ | ||||||
|  | /* | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/offsets.h> | ||||||
|  | #include <grub/symbol.h> | ||||||
|  | #include <grub/machine/kernel.h> | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  |  * GRUB is called from U-Boot as a Linux Kernel type image, which | ||||||
|  |  * means among other things that it always enters in ARM state. | ||||||
|  |  * | ||||||
|  |  * | ||||||
|  |  * Overview of GRUB image layout: | ||||||
|  |  * | ||||||
|  |  * _start: | ||||||
|  |  *              Entry point (1 ARM branch instruction, to "codestart") | ||||||
|  |  * grub_total_module_size: | ||||||
|  |  *              Data field: Size of included module blob | ||||||
|  |  *              (when generated by grub-mkimage) | ||||||
|  |  * codestart: | ||||||
|  |  *              Remainder of statically-linked executable code and data. | ||||||
|  |  * __bss_start: | ||||||
|  |  *              Start of included module blob. | ||||||
|  |  *              Also where global/static variables are located. | ||||||
|  |  * _end: | ||||||
|  |  *              End of bss region (but not necessarily module blob). | ||||||
|  |  * <overflow>: | ||||||
|  |  *              Any part of the module blob that extends beyond _end. | ||||||
|  |  * <modules>: | ||||||
|  |  *              Loadable modules, post relocation. | ||||||
|  |  * <stack>:      | ||||||
|  |  * <heap>: | ||||||
|  |  */ | ||||||
|  | 	 | ||||||
|  | 	.text | ||||||
|  | 	.arm | ||||||
|  | FUNCTION(_start) | ||||||
|  | 	b	codestart | ||||||
|  | 	 | ||||||
|  | 	@ Size of final image integrated module blob - set by grub-mkimage
 | ||||||
|  | 	. = _start + GRUB_KERNEL_MACHINE_TOTAL_MODULE_SIZE | ||||||
|  | VARIABLE(grub_total_module_size) | ||||||
|  | 	.long 	0
 | ||||||
|  | 
 | ||||||
|  | FUNCTION(codestart) | ||||||
|  | 	@ Store context: Machine ID, atags/dtb, ...
 | ||||||
|  | 	@ U-Boot API signature is stored on the U-Boot heap
 | ||||||
|  | 	@ Stack pointer used as start address for signature probing
 | ||||||
|  | 	mov	r12, sp | ||||||
|  | 	ldr	sp, =entry_state | ||||||
|  | 	push	{r4-r12,lr}	@ store U-Boot context (sp in r12)
 | ||||||
|  | 
 | ||||||
|  | 	@ Put kernel parameters aside until we can store them (further down)
 | ||||||
|  | 	mov	r4, r1		@ machine type
 | ||||||
|  | 	mov	r5, r2		@ boot data
 | ||||||
|  | 
 | ||||||
|  | 	@ Modules have been stored as a blob in BSS,
 | ||||||
|  | 	@ they need to be manually relocated to _end or
 | ||||||
|  | 	@ (__bss_start + grub_total_module_size), whichever greater.
 | ||||||
|  | 	bl	uboot_get_real_bss_start	@ r0 = src
 | ||||||
|  | 	ldr	r1, =EXT_C(_end)		@ dst = End of BSS
 | ||||||
|  | 	ldr	r2, grub_total_module_size	@ blob size
 | ||||||
|  | 	add	r3, r0, r2			@ blob end
 | ||||||
|  | 	cmp	r1, r3				@ _end < blob end?
 | ||||||
|  | 	movlt	r1, r3				@ dst = blob end + blob size
 | ||||||
|  | 	 | ||||||
|  | 1:	ldr	r3, [r0], #4 			@ r3 = *src++ 
 | ||||||
|  | 	str	r3, [r1], #4			@ *dst++ = r3 
 | ||||||
|  | 	subs	r2, #4				@ remaining -= 4
 | ||||||
|  | 	bne	1b				@ while remaining != 0
 | ||||||
|  | 	 | ||||||
|  | 	@ Set up a new stack, beyond the end of copied modules.
 | ||||||
|  | 	ldr	r3, =GRUB_KERNEL_MACHINE_STACK_SIZE | ||||||
|  | 	add	r3, r1, r3	@ Place stack beyond end of modules
 | ||||||
|  | 	and	sp, r3, #~0x7	@ Ensure 8-byte alignment
 | ||||||
|  | 
 | ||||||
|  | 	@ Since we _are_ the C run-time, we need to manually zero the BSS
 | ||||||
|  | 	@ region before continuing
 | ||||||
|  | 	bl	uboot_get_real_bss_start	@ zero from here
 | ||||||
|  | 	ldr	r1, =EXT_C(_end)		@ to here
 | ||||||
|  | 	mov	r2, #0 | ||||||
|  | 1:	str	r2, [r0], #4 | ||||||
|  | 	cmp	r0, r1 | ||||||
|  | 	bne	1b | ||||||
|  | 
 | ||||||
|  | 	@ Global variables now accessible - store kernel parameters in memory
 | ||||||
|  | 	ldr     r12, =EXT_C(uboot_machine_type) | ||||||
|  | 	str     r4, [r12] | ||||||
|  | 	ldr     r12, =EXT_C(uboot_boot_data) | ||||||
|  | 	str     r5, [r12] | ||||||
|  | 	 | ||||||
|  | 	b	EXT_C(grub_main) | ||||||
|  | 
 | ||||||
|  | 	/* | ||||||
|  | 	 * __bss_start does not actually point to the start of the runtime | ||||||
|  | 	 * BSS, but rather to the next byte following the preceding data. | ||||||
|  | 	 */ | ||||||
|  | FUNCTION (uboot_get_real_bss_start) | ||||||
|  | 	ldr	r0, =EXT_C(__bss_start)		@ src
 | ||||||
|  | 	tst	r0, #(GRUB_KERNEL_MACHINE_MOD_ALIGN - 1) | ||||||
|  | 	beq	1f | ||||||
|  | 	mvn	r1, #(GRUB_KERNEL_MACHINE_MOD_ALIGN - 1) | ||||||
|  | 	and	r0, r0, r1 | ||||||
|  | 	add	r0, r0, #(GRUB_KERNEL_MACHINE_MOD_ALIGN) | ||||||
|  | 1:	bx	lr | ||||||
|  | 
 | ||||||
|  | 	/* | ||||||
|  | 	 * uboot_syscall(): | ||||||
|  | 	 *   This function is effectively a veneer, so it cannot | ||||||
|  | 	 *   modify the stack or corrupt any registers other than | ||||||
|  | 	 *   r12 (ip). Furthermore it needs to restore r8 for | ||||||
|  | 	 *   U-Boot (Global Data Pointer) and preserve it for Grub. | ||||||
|  | 	 */ | ||||||
|  | FUNCTION(uboot_syscall) | ||||||
|  | 	ldr	ip, =transition_space | ||||||
|  | 	stm	ip, {r8, lr} | ||||||
|  | 	ldr	ip, =gd_backup | ||||||
|  | 	ldr	r8, [ip] | ||||||
|  | 	ldr	ip, =uboot_syscall_ptr | ||||||
|  | 	mov	lr, pc | ||||||
|  | 	ldr	pc, [ip] | ||||||
|  | 	ldr	ip, =gd_backup | ||||||
|  | 	str	r8, [ip] | ||||||
|  | 	ldr	ip, =transition_space | ||||||
|  | 	ldm	ip, {r8, lr} | ||||||
|  | 	bx	lr | ||||||
|  | 	 | ||||||
|  | FUNCTION(uboot_return) | ||||||
|  | 	ldr	sp, =entry_state_end | ||||||
|  | 	pop	{r4-r12, lr} | ||||||
|  | 	mov	sp, r12 | ||||||
|  | 	bx	lr | ||||||
|  | 
 | ||||||
|  | 	 | ||||||
|  | 	.data | ||||||
|  | 	.align	3	@ 8-byte alignment for stack
 | ||||||
|  | @ U-boot context stack space
 | ||||||
|  | entry_state_end:	 | ||||||
|  | 	.long	0	@ r4
 | ||||||
|  | 	.long	0	@ r5
 | ||||||
|  | 	.long	0	@ r6
 | ||||||
|  | 	.long	0	@ r7
 | ||||||
|  | gd_backup:	 | ||||||
|  | 	.long	0	@ r8 - U-Boot global data pointer
 | ||||||
|  | 	.long	0	@ r9
 | ||||||
|  | 	.long	0	@ r10
 | ||||||
|  | 	.long	0	@ r11
 | ||||||
|  | VARIABLE(uboot_search_hint)@ U-Boot stack pointer - 
 | ||||||
|  | 	.long	0	@ also API signature address hint.
 | ||||||
|  | 	.long	0	@ lr
 | ||||||
|  | entry_state:		@ backup for U-Boot context
 | ||||||
|  | 
 | ||||||
|  | @ GRUB context stack space
 | ||||||
|  | transition_space:	 | ||||||
|  | 	.long	0	@ r8
 | ||||||
|  | 	.long	0	@ lr
 | ||||||
|  | 
 | ||||||
|  | VARIABLE(uboot_syscall_ptr) | ||||||
|  | 	.long	0	@
 | ||||||
|  | 
 | ||||||
|  | 	.end | ||||||
							
								
								
									
										112
									
								
								grub-core/kern/uboot/hw.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								grub-core/kern/uboot/hw.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,112 @@ | ||||||
|  | /* hw.c - U-Boot hardware discovery */ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/kernel.h> | ||||||
|  | #include <grub/memory.h> | ||||||
|  | #include <grub/misc.h> | ||||||
|  | #include <grub/mm.h> | ||||||
|  | #include <grub/offsets.h> | ||||||
|  | #include <grub/machine/kernel.h> | ||||||
|  | #include <grub/uboot/disk.h> | ||||||
|  | #include <grub/uboot/uboot.h> | ||||||
|  | 
 | ||||||
|  | grub_addr_t start_of_ram; | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * grub_uboot_probe_memory(): | ||||||
|  |  *   Queries U-Boot for available memory regions. | ||||||
|  |  * | ||||||
|  |  *   Sets up heap near the image in memory and sets up "start_of_ram". | ||||||
|  |  */ | ||||||
|  | void | ||||||
|  | grub_uboot_mm_init (void) | ||||||
|  | { | ||||||
|  |   struct sys_info *si = uboot_get_sys_info (); | ||||||
|  | 
 | ||||||
|  |   grub_mm_init_region ((void *) (grub_modules_get_end () | ||||||
|  | 				 + GRUB_KERNEL_MACHINE_STACK_SIZE), | ||||||
|  | 		       GRUB_KERNEL_MACHINE_HEAP_SIZE); | ||||||
|  | 
 | ||||||
|  |   if (si && (si->mr_no != 0)) | ||||||
|  |     { | ||||||
|  |       int i; | ||||||
|  |       start_of_ram = GRUB_UINT_MAX; | ||||||
|  | 
 | ||||||
|  |       for (i = 0; i < si->mr_no; i++) | ||||||
|  | 	if ((si->mr[i].flags & MR_ATTR_MASK) == MR_ATTR_DRAM) | ||||||
|  | 	  if (si->mr[i].start < start_of_ram) | ||||||
|  | 	    start_of_ram = si->mr[i].start; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * grub_uboot_probe_hardware(): | ||||||
|  |  *    | ||||||
|  |  */ | ||||||
|  | grub_err_t | ||||||
|  | grub_uboot_probe_hardware (void) | ||||||
|  | { | ||||||
|  |   int devcount, i; | ||||||
|  | 
 | ||||||
|  |   devcount = uboot_dev_enum (); | ||||||
|  |   grub_dprintf ("init", "%d devices found\n", devcount); | ||||||
|  | 
 | ||||||
|  |   for (i = 0; i < devcount; i++) | ||||||
|  |     { | ||||||
|  |       struct device_info *devinfo = uboot_dev_get (i); | ||||||
|  | 
 | ||||||
|  |       grub_dprintf ("init", "device handle: %d\n", i); | ||||||
|  |       grub_dprintf ("init", "  cookie\t= 0x%08x\n", | ||||||
|  | 		    (grub_uint32_t) devinfo->cookie); | ||||||
|  | 
 | ||||||
|  |       if (devinfo->type & DEV_TYP_STOR) | ||||||
|  | 	{ | ||||||
|  | 	  grub_dprintf ("init", "  type\t\t= DISK\n"); | ||||||
|  | 	  grub_ubootdisk_register (devinfo, i); | ||||||
|  | 	} | ||||||
|  |       else if (devinfo->type & DEV_TYP_NET) | ||||||
|  | 	{ | ||||||
|  | 	  grub_dprintf ("init", "  type\t\t= NET (not supported yet)\n"); | ||||||
|  | 	} | ||||||
|  |       else | ||||||
|  | 	{ | ||||||
|  | 	  grub_dprintf ("init", "%s: unknown device type", __FUNCTION__); | ||||||
|  | 	} | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | grub_err_t | ||||||
|  | grub_machine_mmap_iterate (grub_memory_hook_t hook, void *hook_data) | ||||||
|  | { | ||||||
|  |   int i; | ||||||
|  |   struct sys_info *si = uboot_get_sys_info (); | ||||||
|  | 
 | ||||||
|  |   if (!si || (si->mr_no < 1)) | ||||||
|  |     return GRUB_ERR_BUG; | ||||||
|  | 
 | ||||||
|  |   /* Iterate and call `hook'.  */ | ||||||
|  |   for (i = 0; i < si->mr_no; i++) | ||||||
|  |     if ((si->mr[i].flags & MR_ATTR_MASK) == MR_ATTR_DRAM) | ||||||
|  |       hook (si->mr[i].start, si->mr[i].size, GRUB_MEMORY_AVAILABLE, | ||||||
|  | 	    hook_data); | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | } | ||||||
							
								
								
									
										171
									
								
								grub-core/kern/uboot/init.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										171
									
								
								grub-core/kern/uboot/init.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,171 @@ | ||||||
|  | /* init.c - generic U-Boot initialization and finalization */ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/env.h> | ||||||
|  | #include <grub/kernel.h> | ||||||
|  | #include <grub/misc.h> | ||||||
|  | #include <grub/mm.h> | ||||||
|  | #include <grub/offsets.h> | ||||||
|  | #include <grub/term.h> | ||||||
|  | #include <grub/time.h> | ||||||
|  | #include <grub/machine/kernel.h> | ||||||
|  | #include <grub/uboot/console.h> | ||||||
|  | #include <grub/uboot/disk.h> | ||||||
|  | #include <grub/uboot/uboot.h> | ||||||
|  | 
 | ||||||
|  | extern char __bss_start[]; | ||||||
|  | extern char _end[]; | ||||||
|  | extern grub_size_t grub_total_module_size; | ||||||
|  | extern int (*uboot_syscall_ptr) (int, int *, ...); | ||||||
|  | 
 | ||||||
|  | grub_addr_t grub_modbase; | ||||||
|  | 
 | ||||||
|  | grub_uint32_t uboot_machine_type; | ||||||
|  | grub_addr_t uboot_boot_data; | ||||||
|  | 
 | ||||||
|  | static unsigned long timer_start; | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | grub_exit (void) | ||||||
|  | { | ||||||
|  |   uboot_return (0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | grub_uint32_t | ||||||
|  | uboot_get_machine_type (void) | ||||||
|  | { | ||||||
|  |   return uboot_machine_type; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | grub_addr_t | ||||||
|  | uboot_get_boot_data (void) | ||||||
|  | { | ||||||
|  |   return uboot_boot_data; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static grub_uint64_t | ||||||
|  | uboot_timer_ms (void) | ||||||
|  | { | ||||||
|  |   return (grub_uint64_t) uboot_get_timer (timer_start); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | grub_machine_init (void) | ||||||
|  | { | ||||||
|  |   grub_addr_t end, real_bss_start; | ||||||
|  |   int ver; | ||||||
|  | 
 | ||||||
|  |   /* First of all - establish connection with U-Boot */ | ||||||
|  |   ver = uboot_api_init (); | ||||||
|  |   if (!ver) | ||||||
|  |     { | ||||||
|  |       /* Don't even have a console to log errors to... */ | ||||||
|  |       grub_exit (); | ||||||
|  |     } | ||||||
|  |   else if (ver > API_SIG_VERSION) | ||||||
|  |     { | ||||||
|  |       /* Try to print an error message */ | ||||||
|  |       uboot_puts ("invalid U-Boot API version\n"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   /*
 | ||||||
|  |    * Modules were relocated to _end, or __bss_start + grub_total_module_size, | ||||||
|  |    * whichever greater. (And __bss_start may not point to actual BSS start...) | ||||||
|  |    */ | ||||||
|  |   real_bss_start = uboot_get_real_bss_start (); | ||||||
|  |   end = real_bss_start + grub_total_module_size; | ||||||
|  |   if (end < (grub_addr_t) _end) | ||||||
|  |     end = (grub_addr_t) _end; | ||||||
|  |   grub_modbase = end; | ||||||
|  | 
 | ||||||
|  |   /* Initialize the console so that GRUB can display messages.  */ | ||||||
|  |   grub_console_init_early (); | ||||||
|  | 
 | ||||||
|  |   /* Enumerate memory and initialize the memory management system. */ | ||||||
|  |   grub_uboot_mm_init (); | ||||||
|  | 
 | ||||||
|  |   grub_dprintf ("init", "__bss_start: 0x%08x, real_bss_start: 0x%08x\n", | ||||||
|  | 		(grub_addr_t) __bss_start, real_bss_start); | ||||||
|  |   grub_dprintf ("init", "end: 0x%08x, _end: 0x%08x\n", | ||||||
|  | 		(grub_addr_t) end, (grub_addr_t) _end); | ||||||
|  |   grub_dprintf ("init", "grub_modbase: %p\n", (void *) grub_modbase); | ||||||
|  |   grub_dprintf ("init", "grub_modules_get_end(): %p\n", | ||||||
|  | 		(void *) grub_modules_get_end ()); | ||||||
|  | 
 | ||||||
|  |   /* Initialise full terminfo support */ | ||||||
|  |   grub_console_init_lately (); | ||||||
|  | 
 | ||||||
|  |   /* Enumerate uboot devices */ | ||||||
|  |   grub_uboot_probe_hardware (); | ||||||
|  | 
 | ||||||
|  |   /* Initialise timer */ | ||||||
|  |   timer_start = uboot_get_timer (0); | ||||||
|  |   grub_install_get_time_ms (uboot_timer_ms); | ||||||
|  | 
 | ||||||
|  |   /* Initialize  */ | ||||||
|  |   grub_ubootdisk_init (); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | grub_machine_fini (void) | ||||||
|  | { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * grub_machine_get_bootlocation(): | ||||||
|  |  *   Called from kern/main.c, which expects a device name (minus parentheses) | ||||||
|  |  *   and a filesystem path back, if any are known. | ||||||
|  |  *   Any returned values must be pointers to dynamically allocated strings. | ||||||
|  |  */ | ||||||
|  | void | ||||||
|  | grub_machine_get_bootlocation (char **device, char **path) | ||||||
|  | { | ||||||
|  |   char *tmp; | ||||||
|  | 
 | ||||||
|  |   tmp = uboot_env_get ("grub_bootdev"); | ||||||
|  |   if (tmp) | ||||||
|  |     { | ||||||
|  |       *device = grub_malloc (grub_strlen (tmp) + 1); | ||||||
|  |       if (*device == NULL) | ||||||
|  | 	return; | ||||||
|  |       grub_strncpy (*device, tmp, grub_strlen (tmp) + 1); | ||||||
|  |     } | ||||||
|  |   else | ||||||
|  |     *device = NULL; | ||||||
|  | 
 | ||||||
|  |   tmp = uboot_env_get ("grub_bootpath"); | ||||||
|  |   if (tmp) | ||||||
|  |     { | ||||||
|  |       *path = grub_malloc (grub_strlen (tmp) + 1); | ||||||
|  |       if (*path == NULL) | ||||||
|  | 	return; | ||||||
|  |       grub_strncpy (*path, tmp, grub_strlen (tmp) + 1); | ||||||
|  |     } | ||||||
|  |   else | ||||||
|  |     *path = NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | grub_uboot_fini (void) | ||||||
|  | { | ||||||
|  |   grub_ubootdisk_fini (); | ||||||
|  |   grub_console_fini (); | ||||||
|  | } | ||||||
							
								
								
									
										363
									
								
								grub-core/kern/uboot/uboot.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										363
									
								
								grub-core/kern/uboot/uboot.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,363 @@ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/misc.h> | ||||||
|  | #include <grub/mm.h> | ||||||
|  | #include <grub/uboot/uboot.h> | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * The main syscall entry point is not reentrant, only one call is | ||||||
|  |  * serviced until finished. | ||||||
|  |  * | ||||||
|  |  * int syscall(int call, int *retval, ...) | ||||||
|  |  * e.g. syscall(1, int *, u_int32_t, u_int32_t, u_int32_t, u_int32_t); | ||||||
|  |  * | ||||||
|  |  * call:	syscall number | ||||||
|  |  * | ||||||
|  |  * retval:	points to the return value placeholder, this is the place the | ||||||
|  |  *		syscall puts its return value, if NULL the caller does not | ||||||
|  |  *		expect a return value | ||||||
|  |  * | ||||||
|  |  * ...		syscall arguments (variable number) | ||||||
|  |  * | ||||||
|  |  * returns:	0 if the call not found, 1 if serviced | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | extern int (*uboot_syscall_ptr) (int, int *, ...); | ||||||
|  | extern int uboot_syscall (int, int *, ...); | ||||||
|  | extern grub_addr_t uboot_search_hint; | ||||||
|  | 
 | ||||||
|  | static struct sys_info uboot_sys_info; | ||||||
|  | static struct mem_region uboot_mem_info[5]; | ||||||
|  | static struct device_info uboot_devices[6]; | ||||||
|  | static int num_devices; | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | uboot_api_init (void) | ||||||
|  | { | ||||||
|  |   struct api_signature *start, *end; | ||||||
|  |   struct api_signature *p; | ||||||
|  | 
 | ||||||
|  |   if (uboot_search_hint) | ||||||
|  |     { | ||||||
|  |       /* Extended search range to work around Trim Slice U-Boot issue */ | ||||||
|  |       start = (struct api_signature *) ((uboot_search_hint & ~0x000fffff) | ||||||
|  | 					- 0x00500000); | ||||||
|  |       end = | ||||||
|  | 	(struct api_signature *) ((grub_addr_t) start + UBOOT_API_SEARCH_LEN - | ||||||
|  | 				  API_SIG_MAGLEN + 0x00500000); | ||||||
|  |     } | ||||||
|  |   else | ||||||
|  |     { | ||||||
|  |       start = 0; | ||||||
|  |       end = (struct api_signature *) (256 * 1024 * 1024); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   /* Structure alignment is (at least) 8 bytes */ | ||||||
|  |   for (p = start; p < end; p = (void *) ((grub_addr_t) p + 8)) | ||||||
|  |     { | ||||||
|  |       if (grub_memcmp (&(p->magic), API_SIG_MAGIC, API_SIG_MAGLEN) == 0) | ||||||
|  | 	{ | ||||||
|  | 	  uboot_syscall_ptr = p->syscall; | ||||||
|  | 	  return p->version; | ||||||
|  | 	} | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* All functions below are wrappers around the uboot_syscall() function */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_getc(int *c) | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | uboot_getc (void) | ||||||
|  | { | ||||||
|  |   int c; | ||||||
|  |   if (!uboot_syscall (API_GETC, NULL, &c)) | ||||||
|  |     return -1; | ||||||
|  | 
 | ||||||
|  |   return c; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_tstc(int *c) | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | uboot_tstc (void) | ||||||
|  | { | ||||||
|  |   int c; | ||||||
|  |   if (!uboot_syscall (API_TSTC, NULL, &c)) | ||||||
|  |     return -1; | ||||||
|  | 
 | ||||||
|  |   return c; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_putc(char *ch) | ||||||
|  |  */ | ||||||
|  | void | ||||||
|  | uboot_putc (int c) | ||||||
|  | { | ||||||
|  |   uboot_syscall (API_PUTC, NULL, &c); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_puts(const char *s) | ||||||
|  |  */ | ||||||
|  | void | ||||||
|  | uboot_puts (const char *s) | ||||||
|  | { | ||||||
|  |   uboot_syscall (API_PUTS, NULL, s); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_reset(void) | ||||||
|  |  */ | ||||||
|  | void | ||||||
|  | uboot_reset (void) | ||||||
|  | { | ||||||
|  |   uboot_syscall (API_RESET, NULL, 0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_get_sys_info(struct sys_info *si) | ||||||
|  |  * | ||||||
|  |  * fill out the sys_info struct containing selected parameters about the | ||||||
|  |  * machine | ||||||
|  |  */ | ||||||
|  | struct sys_info * | ||||||
|  | uboot_get_sys_info (void) | ||||||
|  | { | ||||||
|  |   int retval; | ||||||
|  | 
 | ||||||
|  |   grub_memset (&uboot_sys_info, 0, sizeof (uboot_sys_info)); | ||||||
|  |   grub_memset (&uboot_mem_info, 0, sizeof (uboot_mem_info)); | ||||||
|  |   uboot_sys_info.mr = uboot_mem_info; | ||||||
|  |   uboot_sys_info.mr_no = sizeof (uboot_mem_info) / sizeof (struct mem_region); | ||||||
|  | 
 | ||||||
|  |   if (uboot_syscall (API_GET_SYS_INFO, &retval, &uboot_sys_info)) | ||||||
|  |     if (retval == 0) | ||||||
|  |       return &uboot_sys_info; | ||||||
|  | 
 | ||||||
|  |   return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_udelay(unsigned long *udelay) | ||||||
|  |  */ | ||||||
|  | void | ||||||
|  | uboot_udelay (grub_uint32_t usec) | ||||||
|  | { | ||||||
|  |   uboot_syscall (API_UDELAY, NULL, &usec); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_get_timer(unsigned long *current, unsigned long *base) | ||||||
|  |  */ | ||||||
|  | grub_uint32_t | ||||||
|  | uboot_get_timer (grub_uint32_t base) | ||||||
|  | { | ||||||
|  |   grub_uint32_t current; | ||||||
|  | 
 | ||||||
|  |   if (!uboot_syscall (API_GET_TIMER, NULL, ¤t, &base)) | ||||||
|  |     return 0; | ||||||
|  | 
 | ||||||
|  |   return current; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_dev_enum(struct device_info *) | ||||||
|  |  * | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | uboot_dev_enum (void) | ||||||
|  | { | ||||||
|  |   int max; | ||||||
|  | 
 | ||||||
|  |   grub_memset (&uboot_devices, 0, sizeof (uboot_devices)); | ||||||
|  |   max = sizeof (uboot_devices) / sizeof (struct device_info); | ||||||
|  | 
 | ||||||
|  |   /*
 | ||||||
|  |    * The API_DEV_ENUM call starts a fresh enumeration when passed a | ||||||
|  |    * struct device_info with a NULL cookie, and then depends on having | ||||||
|  |    * the prevoiusly enumerated device cookie "seeded" into the target | ||||||
|  |    * structure. | ||||||
|  |    */ | ||||||
|  |   if (!uboot_syscall (API_DEV_ENUM, NULL, &uboot_devices) | ||||||
|  |       || uboot_devices[0].cookie == NULL) | ||||||
|  |     return 0; | ||||||
|  | 
 | ||||||
|  |   for (num_devices = 1; num_devices < max; num_devices++) | ||||||
|  |     { | ||||||
|  |       uboot_devices[num_devices].cookie = | ||||||
|  | 	uboot_devices[num_devices - 1].cookie; | ||||||
|  |       if (!uboot_syscall (API_DEV_ENUM, NULL, &uboot_devices[num_devices])) | ||||||
|  | 	return 0; | ||||||
|  | 
 | ||||||
|  |       /* When no more devices to enumerate, target cookie set to NULL */ | ||||||
|  |       if (uboot_devices[num_devices].cookie == NULL) | ||||||
|  | 	break; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   return num_devices; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #define VALID_DEV(x) (((x) < num_devices) && ((x) >= 0)) | ||||||
|  | #define OPEN_DEV(x) (VALID_DEV(x) && (uboot_devices[(x)].state == DEV_STA_OPEN)) | ||||||
|  | 
 | ||||||
|  | struct device_info * | ||||||
|  | uboot_dev_get (int handle) | ||||||
|  | { | ||||||
|  |   if (VALID_DEV (handle)) | ||||||
|  |     return &uboot_devices[handle]; | ||||||
|  | 
 | ||||||
|  |   return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_dev_open(struct device_info *) | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | uboot_dev_open (int handle) | ||||||
|  | { | ||||||
|  |   struct device_info *dev; | ||||||
|  |   int retval; | ||||||
|  | 
 | ||||||
|  |   if (!VALID_DEV (handle)) | ||||||
|  |     return -1; | ||||||
|  | 
 | ||||||
|  |   dev = &uboot_devices[handle]; | ||||||
|  | 
 | ||||||
|  |   if (!uboot_syscall (API_DEV_OPEN, &retval, dev)) | ||||||
|  |     return -1; | ||||||
|  | 
 | ||||||
|  |   return retval; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_dev_close(struct device_info *) | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | uboot_dev_close (int handle) | ||||||
|  | { | ||||||
|  |   struct device_info *dev; | ||||||
|  |   int retval; | ||||||
|  | 
 | ||||||
|  |   if (!VALID_DEV (handle)) | ||||||
|  |     return -1; | ||||||
|  | 
 | ||||||
|  |   dev = &uboot_devices[handle]; | ||||||
|  | 
 | ||||||
|  |   if (!uboot_syscall (API_DEV_CLOSE, &retval, dev)) | ||||||
|  |     return -1; | ||||||
|  | 
 | ||||||
|  |   return retval; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_dev_read(struct device_info *di, void *buf,	size_t *len, | ||||||
|  |  *                  unsigned long *start, size_t *act_len) | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | uboot_dev_read (int handle, void *buf, lbasize_t blocks, | ||||||
|  | 		lbastart_t start, lbasize_t * real_blocks) | ||||||
|  | { | ||||||
|  |   struct device_info *dev; | ||||||
|  |   int retval; | ||||||
|  | 
 | ||||||
|  |   if (!OPEN_DEV (handle)) | ||||||
|  |     return -1; | ||||||
|  | 
 | ||||||
|  |   dev = &uboot_devices[handle]; | ||||||
|  | 
 | ||||||
|  |   if (!uboot_syscall (API_DEV_READ, &retval, dev, buf, | ||||||
|  | 		      &blocks, &start, real_blocks)) | ||||||
|  |     return -1; | ||||||
|  | 
 | ||||||
|  |   return retval; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_dev_read(struct device_info *di, void *buf, | ||||||
|  |  *                  size_t *len, size_t *act_len) | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | uboot_dev_recv (int handle, void *buf, int size, int *real_size) | ||||||
|  | { | ||||||
|  |   struct device_info *dev; | ||||||
|  |   int retval; | ||||||
|  | 
 | ||||||
|  |   if (!OPEN_DEV (handle)) | ||||||
|  |     return -1; | ||||||
|  | 
 | ||||||
|  |   dev = &uboot_devices[handle]; | ||||||
|  |   if (!uboot_syscall (API_DEV_READ, &retval, dev, buf, &size, real_size)) | ||||||
|  |     return -1; | ||||||
|  | 
 | ||||||
|  |   return retval; | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Notice: this is for sending network packets only, as U-Boot does not | ||||||
|  |  * support writing to storage at the moment (12.2007) | ||||||
|  |  * | ||||||
|  |  * int API_dev_write(struct device_info *di, void *buf,	int *len) | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | uboot_dev_send (int handle, void *buf, int size) | ||||||
|  | { | ||||||
|  |   struct device_info *dev; | ||||||
|  |   int retval; | ||||||
|  | 
 | ||||||
|  |   if (!OPEN_DEV (handle)) | ||||||
|  |     return -1; | ||||||
|  | 
 | ||||||
|  |   dev = &uboot_devices[handle]; | ||||||
|  |   if (!uboot_syscall (API_DEV_WRITE, &retval, dev, buf, &size)) | ||||||
|  |     return -1; | ||||||
|  | 
 | ||||||
|  |   return retval; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_env_get(const char *name, char **value) | ||||||
|  |  */ | ||||||
|  | char * | ||||||
|  | uboot_env_get (const char *name) | ||||||
|  | { | ||||||
|  |   char *value; | ||||||
|  | 
 | ||||||
|  |   if (!uboot_syscall (API_ENV_GET, NULL, name, &value)) | ||||||
|  |     return NULL; | ||||||
|  | 
 | ||||||
|  |   return value; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_env_set(const char *name, const char *value) | ||||||
|  |  */ | ||||||
|  | void | ||||||
|  | uboot_env_set (const char *name, const char *value) | ||||||
|  | { | ||||||
|  |   uboot_syscall (API_ENV_SET, NULL, name, value); | ||||||
|  | } | ||||||
							
								
								
									
										55
									
								
								grub-core/lib/arm/setjmp.S
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								grub-core/lib/arm/setjmp.S
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,55 @@ | ||||||
|  | /* | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/symbol.h> | ||||||
|  | #include <grub/dl.h> | ||||||
|  | 
 | ||||||
|  | 	.file	"setjmp.S" | ||||||
|  | 	.syntax	unified
 | ||||||
|  | #if !defined (__thumb2__) | ||||||
|  | 	.arm | ||||||
|  | #define ARM(x...)	x | ||||||
|  | #define THUMB(x...) | ||||||
|  | #else | ||||||
|  | 	.thumb | ||||||
|  | #define THUMB(x...)	x | ||||||
|  | #define ARM(x...) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | 	.text | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  |  * int grub_setjmp (grub_jmp_buf env) | ||||||
|  |  */ | ||||||
|  | FUNCTION(grub_setjmp) | ||||||
|  |  THUMB(	mov	ip, sp			) | ||||||
|  |  THUMB(	stm	r0, { r4-r11, ip, lr }	) | ||||||
|  |  ARM(	stm	r0, { r4-r11, sp, lr }	) | ||||||
|  | 	mov	r0, #0 | ||||||
|  | 	bx	lr | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  |  * int grub_longjmp (grub_jmp_buf env, int val) | ||||||
|  |  */ | ||||||
|  | FUNCTION(grub_longjmp) | ||||||
|  |  THUMB(	ldm	r0, { r4-r11, ip, lr }	) | ||||||
|  |  THUMB(	mov	sp, ip			) | ||||||
|  |  ARM(	ldm	r0, { r4-r11, sp, lr }	) | ||||||
|  | 	movs	r0, r1 | ||||||
|  | 	moveq	r0, #1 | ||||||
|  | 	bx	lr | ||||||
							
								
								
									
										45
									
								
								grub-core/lib/dtc/libfdt-grub.diff
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								grub-core/lib/dtc/libfdt-grub.diff
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,45 @@ | ||||||
|  | diff -purN libfdt.orig/fdt_rw.c libfdt/fdt_rw.c
 | ||||||
|  | --- libfdt.orig/fdt_rw.c	2011-05-08 20:45:39.000000000 +0100
 | ||||||
|  | +++ libfdt/fdt_rw.c	2012-10-19 15:33:11.085523185 +0100
 | ||||||
|  | @@ -88,9 +88,9 @@ static int _fdt_rw_check_header(void *fd
 | ||||||
|  |   | ||||||
|  |  #define FDT_RW_CHECK_HEADER(fdt) \ | ||||||
|  |  	{ \ | ||||||
|  | -		int err; \
 | ||||||
|  | -		if ((err = _fdt_rw_check_header(fdt)) != 0) \
 | ||||||
|  | -			return err; \
 | ||||||
|  | +		int macro_err; \
 | ||||||
|  | +		if ((macro_err = _fdt_rw_check_header(fdt)) != 0) \
 | ||||||
|  | +			return macro_err; \
 | ||||||
|  |  	} | ||||||
|  |   | ||||||
|  |  static inline int | ||||||
|  | diff -purN libfdt.orig/libfdt_env.h libfdt/libfdt_env.h
 | ||||||
|  | --- libfdt.orig/libfdt_env.h	2011-05-08 20:45:39.000000000 +0100
 | ||||||
|  | +++ libfdt/libfdt_env.h	2012-10-19 16:13:19.051344173 +0100
 | ||||||
|  | @@ -7,6 +7,9 @@
 | ||||||
|  |  #include <stddef.h> | ||||||
|  |  #include <stdint.h> | ||||||
|  |  #include <string.h> | ||||||
|  | +#pragma GCC diagnostic ignored "-Wcast-align"
 | ||||||
|  | +#pragma GCC diagnostic ignored "-Wsign-compare"
 | ||||||
|  | +typedef grub_addr_t uintptr_t;
 | ||||||
|  |   | ||||||
|  |  #define _B(n)	((unsigned long long)((uint8_t *)&x)[n]) | ||||||
|  |  static inline uint32_t | ||||||
|  | diff -purN libfdt.orig/libfdt_internal.h libfdt/libfdt_internal.h
 | ||||||
|  | --- libfdt.orig/libfdt_internal.h	2011-05-08 20:45:39.000000000 +0100
 | ||||||
|  | +++ libfdt/libfdt_internal.h	2012-10-19 15:33:11.105524731 +0100
 | ||||||
|  | @@ -60,9 +60,9 @@
 | ||||||
|  |   | ||||||
|  |  #define FDT_CHECK_HEADER(fdt) \ | ||||||
|  |  	{ \ | ||||||
|  | -		int err; \
 | ||||||
|  | -		if ((err = fdt_check_header(fdt)) != 0) \
 | ||||||
|  | -			return err; \
 | ||||||
|  | +		int macro_err; \
 | ||||||
|  | +		if ((macro_err = fdt_check_header(fdt)) != 0) \
 | ||||||
|  | +			return macro_err; \
 | ||||||
|  |  	} | ||||||
|  |   | ||||||
|  |  int _fdt_check_node_offset (const void *fdt, int offset); | ||||||
							
								
								
									
										10
									
								
								grub-core/lib/dtc/libfdt/Makefile.libfdt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								grub-core/lib/dtc/libfdt/Makefile.libfdt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,10 @@ | ||||||
|  | # Makefile.libfdt | ||||||
|  | # | ||||||
|  | # This is not a complete Makefile of itself.  Instead, it is designed to | ||||||
|  | # be easily embeddable into other systems of Makefiles. | ||||||
|  | # | ||||||
|  | LIBFDT_soname = libfdt.$(SHAREDLIB_EXT).1 | ||||||
|  | LIBFDT_INCLUDES = fdt.h libfdt.h | ||||||
|  | LIBFDT_VERSION = version.lds | ||||||
|  | LIBFDT_SRCS = fdt.c fdt_ro.c fdt_wip.c fdt_sw.c fdt_rw.c fdt_strerror.c | ||||||
|  | LIBFDT_OBJS = $(LIBFDT_SRCS:%.c=%.o) | ||||||
							
								
								
									
										3
									
								
								grub-core/lib/dtc/libfdt/TODO
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								grub-core/lib/dtc/libfdt/TODO
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,3 @@ | ||||||
|  | - Tree traversal functions | ||||||
|  | - Graft function | ||||||
|  | - Complete libfdt.h documenting comments | ||||||
							
								
								
									
										241
									
								
								grub-core/lib/dtc/libfdt/fdt.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										241
									
								
								grub-core/lib/dtc/libfdt/fdt.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,241 @@ | ||||||
|  | /*
 | ||||||
|  |  * libfdt - Flat Device Tree manipulation | ||||||
|  |  * Copyright (C) 2006 David Gibson, IBM Corporation. | ||||||
|  |  * | ||||||
|  |  * libfdt is dual licensed: you can use it either under the terms of | ||||||
|  |  * the GPL, or the BSD license, at your option. | ||||||
|  |  * | ||||||
|  |  *  a) This library 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 2 of the | ||||||
|  |  *     License, or (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  *     This library 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 this library; if not, write to the Free | ||||||
|  |  *     Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, | ||||||
|  |  *     MA 02110-1301 USA | ||||||
|  |  * | ||||||
|  |  * Alternatively, | ||||||
|  |  * | ||||||
|  |  *  b) 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. | ||||||
|  |  * | ||||||
|  |  *     THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS 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 COPYRIGHT OWNER 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. | ||||||
|  |  */ | ||||||
|  | #include "libfdt_env.h" | ||||||
|  | 
 | ||||||
|  | #include <fdt.h> | ||||||
|  | #include <libfdt.h> | ||||||
|  | 
 | ||||||
|  | #include "libfdt_internal.h" | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_check_header (const void *fdt) | ||||||
|  | { | ||||||
|  |   if (fdt_magic (fdt) == FDT_MAGIC) | ||||||
|  |     { | ||||||
|  |       /* Complete tree */ | ||||||
|  |       if (fdt_version (fdt) < FDT_FIRST_SUPPORTED_VERSION) | ||||||
|  | 	return -FDT_ERR_BADVERSION; | ||||||
|  |       if (fdt_last_comp_version (fdt) > FDT_LAST_SUPPORTED_VERSION) | ||||||
|  | 	return -FDT_ERR_BADVERSION; | ||||||
|  |     } | ||||||
|  |   else if (fdt_magic (fdt) == FDT_SW_MAGIC) | ||||||
|  |     { | ||||||
|  |       /* Unfinished sequential-write blob */ | ||||||
|  |       if (fdt_size_dt_struct (fdt) == 0) | ||||||
|  | 	return -FDT_ERR_BADSTATE; | ||||||
|  |     } | ||||||
|  |   else | ||||||
|  |     { | ||||||
|  |       return -FDT_ERR_BADMAGIC; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const void * | ||||||
|  | fdt_offset_ptr (const void *fdt, int offset, unsigned int len) | ||||||
|  | { | ||||||
|  |   const char *p; | ||||||
|  | 
 | ||||||
|  |   if (fdt_version (fdt) >= 0x11) | ||||||
|  |     if (((offset + len) < offset) | ||||||
|  | 	|| ((offset + len) > fdt_size_dt_struct (fdt))) | ||||||
|  |       return NULL; | ||||||
|  | 
 | ||||||
|  |   p = _fdt_offset_ptr (fdt, offset); | ||||||
|  | 
 | ||||||
|  |   if (p + len < p) | ||||||
|  |     return NULL; | ||||||
|  |   return p; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | uint32_t | ||||||
|  | fdt_next_tag (const void *fdt, int startoffset, int *nextoffset) | ||||||
|  | { | ||||||
|  |   const uint32_t *tagp, *lenp; | ||||||
|  |   uint32_t tag; | ||||||
|  |   int offset = startoffset; | ||||||
|  |   const char *p; | ||||||
|  | 
 | ||||||
|  |   *nextoffset = -FDT_ERR_TRUNCATED; | ||||||
|  |   tagp = fdt_offset_ptr (fdt, offset, FDT_TAGSIZE); | ||||||
|  |   if (!tagp) | ||||||
|  |     return FDT_END;		/* premature end */ | ||||||
|  |   tag = fdt32_to_cpu (*tagp); | ||||||
|  |   offset += FDT_TAGSIZE; | ||||||
|  | 
 | ||||||
|  |   *nextoffset = -FDT_ERR_BADSTRUCTURE; | ||||||
|  |   switch (tag) | ||||||
|  |     { | ||||||
|  |     case FDT_BEGIN_NODE: | ||||||
|  |       /* skip name */ | ||||||
|  |       do | ||||||
|  | 	{ | ||||||
|  | 	  p = fdt_offset_ptr (fdt, offset++, 1); | ||||||
|  | 	} | ||||||
|  |       while (p && (*p != '\0')); | ||||||
|  |       if (!p) | ||||||
|  | 	return FDT_END;		/* premature end */ | ||||||
|  |       break; | ||||||
|  | 
 | ||||||
|  |     case FDT_PROP: | ||||||
|  |       lenp = fdt_offset_ptr (fdt, offset, sizeof (*lenp)); | ||||||
|  |       if (!lenp) | ||||||
|  | 	return FDT_END;		/* premature end */ | ||||||
|  |       /* skip-name offset, length and value */ | ||||||
|  |       offset += sizeof (struct fdt_property) - FDT_TAGSIZE | ||||||
|  | 	+ fdt32_to_cpu (*lenp); | ||||||
|  |       break; | ||||||
|  | 
 | ||||||
|  |     case FDT_END: | ||||||
|  |     case FDT_END_NODE: | ||||||
|  |     case FDT_NOP: | ||||||
|  |       break; | ||||||
|  | 
 | ||||||
|  |     default: | ||||||
|  |       return FDT_END; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   if (!fdt_offset_ptr (fdt, startoffset, offset - startoffset)) | ||||||
|  |     return FDT_END;		/* premature end */ | ||||||
|  | 
 | ||||||
|  |   *nextoffset = FDT_TAGALIGN (offset); | ||||||
|  |   return tag; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | _fdt_check_node_offset (const void *fdt, int offset) | ||||||
|  | { | ||||||
|  |   if ((offset < 0) || (offset % FDT_TAGSIZE) | ||||||
|  |       || (fdt_next_tag (fdt, offset, &offset) != FDT_BEGIN_NODE)) | ||||||
|  |     return -FDT_ERR_BADOFFSET; | ||||||
|  | 
 | ||||||
|  |   return offset; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | _fdt_check_prop_offset (const void *fdt, int offset) | ||||||
|  | { | ||||||
|  |   if ((offset < 0) || (offset % FDT_TAGSIZE) | ||||||
|  |       || (fdt_next_tag (fdt, offset, &offset) != FDT_PROP)) | ||||||
|  |     return -FDT_ERR_BADOFFSET; | ||||||
|  | 
 | ||||||
|  |   return offset; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_next_node (const void *fdt, int offset, int *depth) | ||||||
|  | { | ||||||
|  |   int nextoffset = 0; | ||||||
|  |   uint32_t tag; | ||||||
|  | 
 | ||||||
|  |   if (offset >= 0) | ||||||
|  |     if ((nextoffset = _fdt_check_node_offset (fdt, offset)) < 0) | ||||||
|  |       return nextoffset; | ||||||
|  | 
 | ||||||
|  |   do | ||||||
|  |     { | ||||||
|  |       offset = nextoffset; | ||||||
|  |       tag = fdt_next_tag (fdt, offset, &nextoffset); | ||||||
|  | 
 | ||||||
|  |       switch (tag) | ||||||
|  | 	{ | ||||||
|  | 	case FDT_PROP: | ||||||
|  | 	case FDT_NOP: | ||||||
|  | 	  break; | ||||||
|  | 
 | ||||||
|  | 	case FDT_BEGIN_NODE: | ||||||
|  | 	  if (depth) | ||||||
|  | 	    (*depth)++; | ||||||
|  | 	  break; | ||||||
|  | 
 | ||||||
|  | 	case FDT_END_NODE: | ||||||
|  | 	  if (depth && ((--(*depth)) < 0)) | ||||||
|  | 	    return nextoffset; | ||||||
|  | 	  break; | ||||||
|  | 
 | ||||||
|  | 	case FDT_END: | ||||||
|  | 	  if ((nextoffset >= 0) | ||||||
|  | 	      || ((nextoffset == -FDT_ERR_TRUNCATED) && !depth)) | ||||||
|  | 	    return -FDT_ERR_NOTFOUND; | ||||||
|  | 	  else | ||||||
|  | 	    return nextoffset; | ||||||
|  | 	} | ||||||
|  |     } | ||||||
|  |   while (tag != FDT_BEGIN_NODE); | ||||||
|  | 
 | ||||||
|  |   return offset; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const char * | ||||||
|  | _fdt_find_string (const char *strtab, int tabsize, const char *s) | ||||||
|  | { | ||||||
|  |   int len = strlen (s) + 1; | ||||||
|  |   const char *last = strtab + tabsize - len; | ||||||
|  |   const char *p; | ||||||
|  | 
 | ||||||
|  |   for (p = strtab; p <= last; p++) | ||||||
|  |     if (memcmp (p, s, len) == 0) | ||||||
|  |       return p; | ||||||
|  |   return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_move (const void *fdt, void *buf, int bufsize) | ||||||
|  | { | ||||||
|  |   FDT_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   if (fdt_totalsize (fdt) > bufsize) | ||||||
|  |     return -FDT_ERR_NOSPACE; | ||||||
|  | 
 | ||||||
|  |   memmove (buf, fdt, fdt_totalsize (fdt)); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
							
								
								
									
										64
									
								
								grub-core/lib/dtc/libfdt/fdt.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								grub-core/lib/dtc/libfdt/fdt.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,64 @@ | ||||||
|  | #ifndef _FDT_H | ||||||
|  | #define _FDT_H | ||||||
|  | 
 | ||||||
|  | #ifndef __ASSEMBLY__ | ||||||
|  | 
 | ||||||
|  | struct fdt_header | ||||||
|  | { | ||||||
|  |   uint32_t magic;		/* magic word FDT_MAGIC */ | ||||||
|  |   uint32_t totalsize;		/* total size of DT block */ | ||||||
|  |   uint32_t off_dt_struct;	/* offset to structure */ | ||||||
|  |   uint32_t off_dt_strings;	/* offset to strings */ | ||||||
|  |   uint32_t off_mem_rsvmap;	/* offset to memory reserve map */ | ||||||
|  |   uint32_t version;		/* format version */ | ||||||
|  |   uint32_t last_comp_version;	/* last compatible version */ | ||||||
|  | 
 | ||||||
|  |   /* version 2 fields below */ | ||||||
|  |   uint32_t boot_cpuid_phys;	/* Which physical CPU id we're
 | ||||||
|  | 				   booting on */ | ||||||
|  |   /* version 3 fields below */ | ||||||
|  |   uint32_t size_dt_strings;	/* size of the strings block */ | ||||||
|  | 
 | ||||||
|  |   /* version 17 fields below */ | ||||||
|  |   uint32_t size_dt_struct;	/* size of the structure block */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct fdt_reserve_entry | ||||||
|  | { | ||||||
|  |   uint64_t address; | ||||||
|  |   uint64_t size; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct fdt_node_header | ||||||
|  | { | ||||||
|  |   uint32_t tag; | ||||||
|  |   char name[0]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct fdt_property | ||||||
|  | { | ||||||
|  |   uint32_t tag; | ||||||
|  |   uint32_t len; | ||||||
|  |   uint32_t nameoff; | ||||||
|  |   char data[0]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #endif /* !__ASSEMBLY */ | ||||||
|  | 
 | ||||||
|  | #define FDT_MAGIC	0xd00dfeed	/* 4: version, 4: total size */ | ||||||
|  | #define FDT_TAGSIZE	sizeof(uint32_t) | ||||||
|  | 
 | ||||||
|  | #define FDT_BEGIN_NODE	0x1	/* Start node: full name */ | ||||||
|  | #define FDT_END_NODE	0x2	/* End node */ | ||||||
|  | #define FDT_PROP	0x3	/* Property: name off, | ||||||
|  | 				   size, content */ | ||||||
|  | #define FDT_NOP		0x4	/* nop */ | ||||||
|  | #define FDT_END		0x9 | ||||||
|  | 
 | ||||||
|  | #define FDT_V1_SIZE	(7*sizeof(uint32_t)) | ||||||
|  | #define FDT_V2_SIZE	(FDT_V1_SIZE + sizeof(uint32_t)) | ||||||
|  | #define FDT_V3_SIZE	(FDT_V2_SIZE + sizeof(uint32_t)) | ||||||
|  | #define FDT_V16_SIZE	FDT_V3_SIZE | ||||||
|  | #define FDT_V17_SIZE	(FDT_V16_SIZE + sizeof(uint32_t)) | ||||||
|  | 
 | ||||||
|  | #endif /* _FDT_H */ | ||||||
							
								
								
									
										608
									
								
								grub-core/lib/dtc/libfdt/fdt_ro.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										608
									
								
								grub-core/lib/dtc/libfdt/fdt_ro.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,608 @@ | ||||||
|  | /*
 | ||||||
|  |  * libfdt - Flat Device Tree manipulation | ||||||
|  |  * Copyright (C) 2006 David Gibson, IBM Corporation. | ||||||
|  |  * | ||||||
|  |  * libfdt is dual licensed: you can use it either under the terms of | ||||||
|  |  * the GPL, or the BSD license, at your option. | ||||||
|  |  * | ||||||
|  |  *  a) This library 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 2 of the | ||||||
|  |  *     License, or (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  *     This library 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 this library; if not, write to the Free | ||||||
|  |  *     Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, | ||||||
|  |  *     MA 02110-1301 USA | ||||||
|  |  * | ||||||
|  |  * Alternatively, | ||||||
|  |  * | ||||||
|  |  *  b) 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. | ||||||
|  |  * | ||||||
|  |  *     THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS 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 COPYRIGHT OWNER 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. | ||||||
|  |  */ | ||||||
|  | #include "libfdt_env.h" | ||||||
|  | 
 | ||||||
|  | #include <fdt.h> | ||||||
|  | #include <libfdt.h> | ||||||
|  | 
 | ||||||
|  | #include "libfdt_internal.h" | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _fdt_nodename_eq (const void *fdt, int offset, const char *s, int len) | ||||||
|  | { | ||||||
|  |   const char *p = fdt_offset_ptr (fdt, offset + FDT_TAGSIZE, len + 1); | ||||||
|  | 
 | ||||||
|  |   if (!p) | ||||||
|  |     /* short match */ | ||||||
|  |     return 0; | ||||||
|  | 
 | ||||||
|  |   if (memcmp (p, s, len) != 0) | ||||||
|  |     return 0; | ||||||
|  | 
 | ||||||
|  |   if (p[len] == '\0') | ||||||
|  |     return 1; | ||||||
|  |   else if (!memchr (s, '@', len) && (p[len] == '@')) | ||||||
|  |     return 1; | ||||||
|  |   else | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const char * | ||||||
|  | fdt_string (const void *fdt, int stroffset) | ||||||
|  | { | ||||||
|  |   return (const char *) fdt + fdt_off_dt_strings (fdt) + stroffset; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _fdt_string_eq (const void *fdt, int stroffset, const char *s, int len) | ||||||
|  | { | ||||||
|  |   const char *p = fdt_string (fdt, stroffset); | ||||||
|  | 
 | ||||||
|  |   return (strlen (p) == len) && (memcmp (p, s, len) == 0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_get_mem_rsv (const void *fdt, int n, uint64_t * address, uint64_t * size) | ||||||
|  | { | ||||||
|  |   FDT_CHECK_HEADER (fdt); | ||||||
|  |   *address = fdt64_to_cpu (_fdt_mem_rsv (fdt, n)->address); | ||||||
|  |   *size = fdt64_to_cpu (_fdt_mem_rsv (fdt, n)->size); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_num_mem_rsv (const void *fdt) | ||||||
|  | { | ||||||
|  |   int i = 0; | ||||||
|  | 
 | ||||||
|  |   while (fdt64_to_cpu (_fdt_mem_rsv (fdt, i)->size) != 0) | ||||||
|  |     i++; | ||||||
|  |   return i; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _nextprop (const void *fdt, int offset) | ||||||
|  | { | ||||||
|  |   uint32_t tag; | ||||||
|  |   int nextoffset; | ||||||
|  | 
 | ||||||
|  |   do | ||||||
|  |     { | ||||||
|  |       tag = fdt_next_tag (fdt, offset, &nextoffset); | ||||||
|  | 
 | ||||||
|  |       switch (tag) | ||||||
|  | 	{ | ||||||
|  | 	case FDT_END: | ||||||
|  | 	  if (nextoffset >= 0) | ||||||
|  | 	    return -FDT_ERR_BADSTRUCTURE; | ||||||
|  | 	  else | ||||||
|  | 	    return nextoffset; | ||||||
|  | 
 | ||||||
|  | 	case FDT_PROP: | ||||||
|  | 	  return offset; | ||||||
|  | 	} | ||||||
|  |       offset = nextoffset; | ||||||
|  |     } | ||||||
|  |   while (tag == FDT_NOP); | ||||||
|  | 
 | ||||||
|  |   return -FDT_ERR_NOTFOUND; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_subnode_offset_namelen (const void *fdt, int offset, | ||||||
|  | 			    const char *name, int namelen) | ||||||
|  | { | ||||||
|  |   int depth; | ||||||
|  | 
 | ||||||
|  |   FDT_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   for (depth = 0; | ||||||
|  |        (offset >= 0) && (depth >= 0); | ||||||
|  |        offset = fdt_next_node (fdt, offset, &depth)) | ||||||
|  |     if ((depth == 1) && _fdt_nodename_eq (fdt, offset, name, namelen)) | ||||||
|  |       return offset; | ||||||
|  | 
 | ||||||
|  |   if (depth < 0) | ||||||
|  |     return -FDT_ERR_NOTFOUND; | ||||||
|  |   return offset;		/* error */ | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_subnode_offset (const void *fdt, int parentoffset, const char *name) | ||||||
|  | { | ||||||
|  |   return fdt_subnode_offset_namelen (fdt, parentoffset, name, strlen (name)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_path_offset (const void *fdt, const char *path) | ||||||
|  | { | ||||||
|  |   const char *end = path + strlen (path); | ||||||
|  |   const char *p = path; | ||||||
|  |   int offset = 0; | ||||||
|  | 
 | ||||||
|  |   FDT_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   /* see if we have an alias */ | ||||||
|  |   if (*path != '/') | ||||||
|  |     { | ||||||
|  |       const char *q = strchr (path, '/'); | ||||||
|  | 
 | ||||||
|  |       if (!q) | ||||||
|  | 	q = end; | ||||||
|  | 
 | ||||||
|  |       p = fdt_get_alias_namelen (fdt, p, q - p); | ||||||
|  |       if (!p) | ||||||
|  | 	return -FDT_ERR_BADPATH; | ||||||
|  |       offset = fdt_path_offset (fdt, p); | ||||||
|  | 
 | ||||||
|  |       p = q; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   while (*p) | ||||||
|  |     { | ||||||
|  |       const char *q; | ||||||
|  | 
 | ||||||
|  |       while (*p == '/') | ||||||
|  | 	p++; | ||||||
|  |       if (!*p) | ||||||
|  | 	return offset; | ||||||
|  |       q = strchr (p, '/'); | ||||||
|  |       if (!q) | ||||||
|  | 	q = end; | ||||||
|  | 
 | ||||||
|  |       offset = fdt_subnode_offset_namelen (fdt, offset, p, q - p); | ||||||
|  |       if (offset < 0) | ||||||
|  | 	return offset; | ||||||
|  | 
 | ||||||
|  |       p = q; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   return offset; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const char * | ||||||
|  | fdt_get_name (const void *fdt, int nodeoffset, int *len) | ||||||
|  | { | ||||||
|  |   const struct fdt_node_header *nh = _fdt_offset_ptr (fdt, nodeoffset); | ||||||
|  |   int err; | ||||||
|  | 
 | ||||||
|  |   if (((err = fdt_check_header (fdt)) != 0) | ||||||
|  |       || ((err = _fdt_check_node_offset (fdt, nodeoffset)) < 0)) | ||||||
|  |     goto fail; | ||||||
|  | 
 | ||||||
|  |   if (len) | ||||||
|  |     *len = strlen (nh->name); | ||||||
|  | 
 | ||||||
|  |   return nh->name; | ||||||
|  | 
 | ||||||
|  | fail: | ||||||
|  |   if (len) | ||||||
|  |     *len = err; | ||||||
|  |   return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_first_property_offset (const void *fdt, int nodeoffset) | ||||||
|  | { | ||||||
|  |   int offset; | ||||||
|  | 
 | ||||||
|  |   if ((offset = _fdt_check_node_offset (fdt, nodeoffset)) < 0) | ||||||
|  |     return offset; | ||||||
|  | 
 | ||||||
|  |   return _nextprop (fdt, offset); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_next_property_offset (const void *fdt, int offset) | ||||||
|  | { | ||||||
|  |   if ((offset = _fdt_check_prop_offset (fdt, offset)) < 0) | ||||||
|  |     return offset; | ||||||
|  | 
 | ||||||
|  |   return _nextprop (fdt, offset); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const struct fdt_property * | ||||||
|  | fdt_get_property_by_offset (const void *fdt, int offset, int *lenp) | ||||||
|  | { | ||||||
|  |   int err; | ||||||
|  |   const struct fdt_property *prop; | ||||||
|  | 
 | ||||||
|  |   if ((err = _fdt_check_prop_offset (fdt, offset)) < 0) | ||||||
|  |     { | ||||||
|  |       if (lenp) | ||||||
|  | 	*lenp = err; | ||||||
|  |       return NULL; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   prop = _fdt_offset_ptr (fdt, offset); | ||||||
|  | 
 | ||||||
|  |   if (lenp) | ||||||
|  |     *lenp = fdt32_to_cpu (prop->len); | ||||||
|  | 
 | ||||||
|  |   return prop; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const struct fdt_property * | ||||||
|  | fdt_get_property_namelen (const void *fdt, | ||||||
|  | 			  int offset, | ||||||
|  | 			  const char *name, int namelen, int *lenp) | ||||||
|  | { | ||||||
|  |   for (offset = fdt_first_property_offset (fdt, offset); | ||||||
|  |        (offset >= 0); (offset = fdt_next_property_offset (fdt, offset))) | ||||||
|  |     { | ||||||
|  |       const struct fdt_property *prop; | ||||||
|  | 
 | ||||||
|  |       if (!(prop = fdt_get_property_by_offset (fdt, offset, lenp))) | ||||||
|  | 	{ | ||||||
|  | 	  offset = -FDT_ERR_INTERNAL; | ||||||
|  | 	  break; | ||||||
|  | 	} | ||||||
|  |       if (_fdt_string_eq (fdt, fdt32_to_cpu (prop->nameoff), name, namelen)) | ||||||
|  | 	return prop; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   if (lenp) | ||||||
|  |     *lenp = offset; | ||||||
|  |   return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const struct fdt_property * | ||||||
|  | fdt_get_property (const void *fdt, | ||||||
|  | 		  int nodeoffset, const char *name, int *lenp) | ||||||
|  | { | ||||||
|  |   return fdt_get_property_namelen (fdt, nodeoffset, name, | ||||||
|  | 				   strlen (name), lenp); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const void * | ||||||
|  | fdt_getprop_namelen (const void *fdt, int nodeoffset, | ||||||
|  | 		     const char *name, int namelen, int *lenp) | ||||||
|  | { | ||||||
|  |   const struct fdt_property *prop; | ||||||
|  | 
 | ||||||
|  |   prop = fdt_get_property_namelen (fdt, nodeoffset, name, namelen, lenp); | ||||||
|  |   if (!prop) | ||||||
|  |     return NULL; | ||||||
|  | 
 | ||||||
|  |   return prop->data; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const void * | ||||||
|  | fdt_getprop_by_offset (const void *fdt, int offset, | ||||||
|  | 		       const char **namep, int *lenp) | ||||||
|  | { | ||||||
|  |   const struct fdt_property *prop; | ||||||
|  | 
 | ||||||
|  |   prop = fdt_get_property_by_offset (fdt, offset, lenp); | ||||||
|  |   if (!prop) | ||||||
|  |     return NULL; | ||||||
|  |   if (namep) | ||||||
|  |     *namep = fdt_string (fdt, fdt32_to_cpu (prop->nameoff)); | ||||||
|  |   return prop->data; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const void * | ||||||
|  | fdt_getprop (const void *fdt, int nodeoffset, const char *name, int *lenp) | ||||||
|  | { | ||||||
|  |   return fdt_getprop_namelen (fdt, nodeoffset, name, strlen (name), lenp); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | uint32_t | ||||||
|  | fdt_get_phandle (const void *fdt, int nodeoffset) | ||||||
|  | { | ||||||
|  |   const uint32_t *php; | ||||||
|  |   int len; | ||||||
|  | 
 | ||||||
|  |   /* FIXME: This is a bit sub-optimal, since we potentially scan
 | ||||||
|  |    * over all the properties twice. */ | ||||||
|  |   php = fdt_getprop (fdt, nodeoffset, "phandle", &len); | ||||||
|  |   if (!php || (len != sizeof (*php))) | ||||||
|  |     { | ||||||
|  |       php = fdt_getprop (fdt, nodeoffset, "linux,phandle", &len); | ||||||
|  |       if (!php || (len != sizeof (*php))) | ||||||
|  | 	return 0; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   return fdt32_to_cpu (*php); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const char * | ||||||
|  | fdt_get_alias_namelen (const void *fdt, const char *name, int namelen) | ||||||
|  | { | ||||||
|  |   int aliasoffset; | ||||||
|  | 
 | ||||||
|  |   aliasoffset = fdt_path_offset (fdt, "/aliases"); | ||||||
|  |   if (aliasoffset < 0) | ||||||
|  |     return NULL; | ||||||
|  | 
 | ||||||
|  |   return fdt_getprop_namelen (fdt, aliasoffset, name, namelen, NULL); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const char * | ||||||
|  | fdt_get_alias (const void *fdt, const char *name) | ||||||
|  | { | ||||||
|  |   return fdt_get_alias_namelen (fdt, name, strlen (name)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_get_path (const void *fdt, int nodeoffset, char *buf, int buflen) | ||||||
|  | { | ||||||
|  |   int pdepth = 0, p = 0; | ||||||
|  |   int offset, depth, namelen; | ||||||
|  |   const char *name; | ||||||
|  | 
 | ||||||
|  |   FDT_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   if (buflen < 2) | ||||||
|  |     return -FDT_ERR_NOSPACE; | ||||||
|  | 
 | ||||||
|  |   for (offset = 0, depth = 0; | ||||||
|  |        (offset >= 0) && (offset <= nodeoffset); | ||||||
|  |        offset = fdt_next_node (fdt, offset, &depth)) | ||||||
|  |     { | ||||||
|  |       while (pdepth > depth) | ||||||
|  | 	{ | ||||||
|  | 	  do | ||||||
|  | 	    { | ||||||
|  | 	      p--; | ||||||
|  | 	    } | ||||||
|  | 	  while (buf[p - 1] != '/'); | ||||||
|  | 	  pdepth--; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  |       if (pdepth >= depth) | ||||||
|  | 	{ | ||||||
|  | 	  name = fdt_get_name (fdt, offset, &namelen); | ||||||
|  | 	  if (!name) | ||||||
|  | 	    return namelen; | ||||||
|  | 	  if ((p + namelen + 1) <= buflen) | ||||||
|  | 	    { | ||||||
|  | 	      memcpy (buf + p, name, namelen); | ||||||
|  | 	      p += namelen; | ||||||
|  | 	      buf[p++] = '/'; | ||||||
|  | 	      pdepth++; | ||||||
|  | 	    } | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  |       if (offset == nodeoffset) | ||||||
|  | 	{ | ||||||
|  | 	  if (pdepth < (depth + 1)) | ||||||
|  | 	    return -FDT_ERR_NOSPACE; | ||||||
|  | 
 | ||||||
|  | 	  if (p > 1)		/* special case so that root path is "/", not "" */ | ||||||
|  | 	    p--; | ||||||
|  | 	  buf[p] = '\0'; | ||||||
|  | 	  return 0; | ||||||
|  | 	} | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   if ((offset == -FDT_ERR_NOTFOUND) || (offset >= 0)) | ||||||
|  |     return -FDT_ERR_BADOFFSET; | ||||||
|  |   else if (offset == -FDT_ERR_BADOFFSET) | ||||||
|  |     return -FDT_ERR_BADSTRUCTURE; | ||||||
|  | 
 | ||||||
|  |   return offset;		/* error from fdt_next_node() */ | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_supernode_atdepth_offset (const void *fdt, int nodeoffset, | ||||||
|  | 			      int supernodedepth, int *nodedepth) | ||||||
|  | { | ||||||
|  |   int offset, depth; | ||||||
|  |   int supernodeoffset = -FDT_ERR_INTERNAL; | ||||||
|  | 
 | ||||||
|  |   FDT_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   if (supernodedepth < 0) | ||||||
|  |     return -FDT_ERR_NOTFOUND; | ||||||
|  | 
 | ||||||
|  |   for (offset = 0, depth = 0; | ||||||
|  |        (offset >= 0) && (offset <= nodeoffset); | ||||||
|  |        offset = fdt_next_node (fdt, offset, &depth)) | ||||||
|  |     { | ||||||
|  |       if (depth == supernodedepth) | ||||||
|  | 	supernodeoffset = offset; | ||||||
|  | 
 | ||||||
|  |       if (offset == nodeoffset) | ||||||
|  | 	{ | ||||||
|  | 	  if (nodedepth) | ||||||
|  | 	    *nodedepth = depth; | ||||||
|  | 
 | ||||||
|  | 	  if (supernodedepth > depth) | ||||||
|  | 	    return -FDT_ERR_NOTFOUND; | ||||||
|  | 	  else | ||||||
|  | 	    return supernodeoffset; | ||||||
|  | 	} | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   if ((offset == -FDT_ERR_NOTFOUND) || (offset >= 0)) | ||||||
|  |     return -FDT_ERR_BADOFFSET; | ||||||
|  |   else if (offset == -FDT_ERR_BADOFFSET) | ||||||
|  |     return -FDT_ERR_BADSTRUCTURE; | ||||||
|  | 
 | ||||||
|  |   return offset;		/* error from fdt_next_node() */ | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_node_depth (const void *fdt, int nodeoffset) | ||||||
|  | { | ||||||
|  |   int nodedepth; | ||||||
|  |   int err; | ||||||
|  | 
 | ||||||
|  |   err = fdt_supernode_atdepth_offset (fdt, nodeoffset, 0, &nodedepth); | ||||||
|  |   if (err) | ||||||
|  |     return (err < 0) ? err : -FDT_ERR_INTERNAL; | ||||||
|  |   return nodedepth; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_parent_offset (const void *fdt, int nodeoffset) | ||||||
|  | { | ||||||
|  |   int nodedepth = fdt_node_depth (fdt, nodeoffset); | ||||||
|  | 
 | ||||||
|  |   if (nodedepth < 0) | ||||||
|  |     return nodedepth; | ||||||
|  |   return fdt_supernode_atdepth_offset (fdt, nodeoffset, nodedepth - 1, NULL); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_node_offset_by_prop_value (const void *fdt, int startoffset, | ||||||
|  | 			       const char *propname, | ||||||
|  | 			       const void *propval, int proplen) | ||||||
|  | { | ||||||
|  |   int offset; | ||||||
|  |   const void *val; | ||||||
|  |   int len; | ||||||
|  | 
 | ||||||
|  |   FDT_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   /* FIXME: The algorithm here is pretty horrible: we scan each
 | ||||||
|  |    * property of a node in fdt_getprop(), then if that didn't | ||||||
|  |    * find what we want, we scan over them again making our way | ||||||
|  |    * to the next node.  Still it's the easiest to implement | ||||||
|  |    * approach; performance can come later. */ | ||||||
|  |   for (offset = fdt_next_node (fdt, startoffset, NULL); | ||||||
|  |        offset >= 0; offset = fdt_next_node (fdt, offset, NULL)) | ||||||
|  |     { | ||||||
|  |       val = fdt_getprop (fdt, offset, propname, &len); | ||||||
|  |       if (val && (len == proplen) && (memcmp (val, propval, len) == 0)) | ||||||
|  | 	return offset; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   return offset;		/* error from fdt_next_node() */ | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_node_offset_by_phandle (const void *fdt, uint32_t phandle) | ||||||
|  | { | ||||||
|  |   int offset; | ||||||
|  | 
 | ||||||
|  |   if ((phandle == 0) || (phandle == -1)) | ||||||
|  |     return -FDT_ERR_BADPHANDLE; | ||||||
|  | 
 | ||||||
|  |   FDT_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   /* FIXME: The algorithm here is pretty horrible: we
 | ||||||
|  |    * potentially scan each property of a node in | ||||||
|  |    * fdt_get_phandle(), then if that didn't find what | ||||||
|  |    * we want, we scan over them again making our way to the next | ||||||
|  |    * node.  Still it's the easiest to implement approach; | ||||||
|  |    * performance can come later. */ | ||||||
|  |   for (offset = fdt_next_node (fdt, -1, NULL); | ||||||
|  |        offset >= 0; offset = fdt_next_node (fdt, offset, NULL)) | ||||||
|  |     { | ||||||
|  |       if (fdt_get_phandle (fdt, offset) == phandle) | ||||||
|  | 	return offset; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   return offset;		/* error from fdt_next_node() */ | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _fdt_stringlist_contains (const char *strlist, int listlen, const char *str) | ||||||
|  | { | ||||||
|  |   int len = strlen (str); | ||||||
|  |   const char *p; | ||||||
|  | 
 | ||||||
|  |   while (listlen >= len) | ||||||
|  |     { | ||||||
|  |       if (memcmp (str, strlist, len + 1) == 0) | ||||||
|  | 	return 1; | ||||||
|  |       p = memchr (strlist, '\0', listlen); | ||||||
|  |       if (!p) | ||||||
|  | 	return 0;		/* malformed strlist.. */ | ||||||
|  |       listlen -= (p - strlist) + 1; | ||||||
|  |       strlist = p + 1; | ||||||
|  |     } | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_node_check_compatible (const void *fdt, int nodeoffset, | ||||||
|  | 			   const char *compatible) | ||||||
|  | { | ||||||
|  |   const void *prop; | ||||||
|  |   int len; | ||||||
|  | 
 | ||||||
|  |   prop = fdt_getprop (fdt, nodeoffset, "compatible", &len); | ||||||
|  |   if (!prop) | ||||||
|  |     return len; | ||||||
|  |   if (_fdt_stringlist_contains (prop, len, compatible)) | ||||||
|  |     return 0; | ||||||
|  |   else | ||||||
|  |     return 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_node_offset_by_compatible (const void *fdt, int startoffset, | ||||||
|  | 			       const char *compatible) | ||||||
|  | { | ||||||
|  |   int offset, err; | ||||||
|  | 
 | ||||||
|  |   FDT_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   /* FIXME: The algorithm here is pretty horrible: we scan each
 | ||||||
|  |    * property of a node in fdt_node_check_compatible(), then if | ||||||
|  |    * that didn't find what we want, we scan over them again | ||||||
|  |    * making our way to the next node.  Still it's the easiest to | ||||||
|  |    * implement approach; performance can come later. */ | ||||||
|  |   for (offset = fdt_next_node (fdt, startoffset, NULL); | ||||||
|  |        offset >= 0; offset = fdt_next_node (fdt, offset, NULL)) | ||||||
|  |     { | ||||||
|  |       err = fdt_node_check_compatible (fdt, offset, compatible); | ||||||
|  |       if ((err < 0) && (err != -FDT_ERR_NOTFOUND)) | ||||||
|  | 	return err; | ||||||
|  |       else if (err == 0) | ||||||
|  | 	return offset; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   return offset;		/* error from fdt_next_node() */ | ||||||
|  | } | ||||||
							
								
								
									
										490
									
								
								grub-core/lib/dtc/libfdt/fdt_rw.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										490
									
								
								grub-core/lib/dtc/libfdt/fdt_rw.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,490 @@ | ||||||
|  | /*
 | ||||||
|  |  * libfdt - Flat Device Tree manipulation | ||||||
|  |  * Copyright (C) 2006 David Gibson, IBM Corporation. | ||||||
|  |  * | ||||||
|  |  * libfdt is dual licensed: you can use it either under the terms of | ||||||
|  |  * the GPL, or the BSD license, at your option. | ||||||
|  |  * | ||||||
|  |  *  a) This library 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 2 of the | ||||||
|  |  *     License, or (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  *     This library 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 this library; if not, write to the Free | ||||||
|  |  *     Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, | ||||||
|  |  *     MA 02110-1301 USA | ||||||
|  |  * | ||||||
|  |  * Alternatively, | ||||||
|  |  * | ||||||
|  |  *  b) 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. | ||||||
|  |  * | ||||||
|  |  *     THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS 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 COPYRIGHT OWNER 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. | ||||||
|  |  */ | ||||||
|  | #include "libfdt_env.h" | ||||||
|  | 
 | ||||||
|  | #include <fdt.h> | ||||||
|  | #include <libfdt.h> | ||||||
|  | 
 | ||||||
|  | #include "libfdt_internal.h" | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _fdt_blocks_misordered (const void *fdt, int mem_rsv_size, int struct_size) | ||||||
|  | { | ||||||
|  |   return (fdt_off_mem_rsvmap (fdt) < | ||||||
|  | 	  FDT_ALIGN (sizeof (struct fdt_header), 8)) | ||||||
|  |     || (fdt_off_dt_struct (fdt) < (fdt_off_mem_rsvmap (fdt) + mem_rsv_size)) | ||||||
|  |     || (fdt_off_dt_strings (fdt) < (fdt_off_dt_struct (fdt) + struct_size)) | ||||||
|  |     || (fdt_totalsize (fdt) < | ||||||
|  | 	(fdt_off_dt_strings (fdt) + fdt_size_dt_strings (fdt))); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _fdt_rw_check_header (void *fdt) | ||||||
|  | { | ||||||
|  |   FDT_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   if (fdt_version (fdt) < 17) | ||||||
|  |     return -FDT_ERR_BADVERSION; | ||||||
|  |   if (_fdt_blocks_misordered (fdt, sizeof (struct fdt_reserve_entry), | ||||||
|  | 			      fdt_size_dt_struct (fdt))) | ||||||
|  |     return -FDT_ERR_BADLAYOUT; | ||||||
|  |   if (fdt_version (fdt) > 17) | ||||||
|  |     fdt_set_version (fdt, 17); | ||||||
|  | 
 | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #define FDT_RW_CHECK_HEADER(fdt) \ | ||||||
|  | 	{ \ | ||||||
|  | 		int err; \ | ||||||
|  | 		if ((err = _fdt_rw_check_header(fdt)) != 0) \ | ||||||
|  | 			return err; \ | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | static inline int | ||||||
|  | _fdt_data_size (void *fdt) | ||||||
|  | { | ||||||
|  |   return fdt_off_dt_strings (fdt) + fdt_size_dt_strings (fdt); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _fdt_splice (void *fdt, void *splicepoint, int oldlen, int newlen) | ||||||
|  | { | ||||||
|  |   char *p = splicepoint; | ||||||
|  |   char *end = (char *) fdt + _fdt_data_size (fdt); | ||||||
|  | 
 | ||||||
|  |   if (((p + oldlen) < p) || ((p + oldlen) > end)) | ||||||
|  |     return -FDT_ERR_BADOFFSET; | ||||||
|  |   if ((end - oldlen + newlen) > ((char *) fdt + fdt_totalsize (fdt))) | ||||||
|  |     return -FDT_ERR_NOSPACE; | ||||||
|  |   memmove (p + newlen, p + oldlen, end - p - oldlen); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _fdt_splice_mem_rsv (void *fdt, struct fdt_reserve_entry *p, | ||||||
|  | 		     int oldn, int newn) | ||||||
|  | { | ||||||
|  |   int delta = (newn - oldn) * sizeof (*p); | ||||||
|  |   int err; | ||||||
|  |   err = _fdt_splice (fdt, p, oldn * sizeof (*p), newn * sizeof (*p)); | ||||||
|  |   if (err) | ||||||
|  |     return err; | ||||||
|  |   fdt_set_off_dt_struct (fdt, fdt_off_dt_struct (fdt) + delta); | ||||||
|  |   fdt_set_off_dt_strings (fdt, fdt_off_dt_strings (fdt) + delta); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _fdt_splice_struct (void *fdt, void *p, int oldlen, int newlen) | ||||||
|  | { | ||||||
|  |   int delta = newlen - oldlen; | ||||||
|  |   int err; | ||||||
|  | 
 | ||||||
|  |   if ((err = _fdt_splice (fdt, p, oldlen, newlen))) | ||||||
|  |     return err; | ||||||
|  | 
 | ||||||
|  |   fdt_set_size_dt_struct (fdt, fdt_size_dt_struct (fdt) + delta); | ||||||
|  |   fdt_set_off_dt_strings (fdt, fdt_off_dt_strings (fdt) + delta); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _fdt_splice_string (void *fdt, int newlen) | ||||||
|  | { | ||||||
|  |   void *p = (char *) fdt | ||||||
|  |     + fdt_off_dt_strings (fdt) + fdt_size_dt_strings (fdt); | ||||||
|  |   int err; | ||||||
|  | 
 | ||||||
|  |   if ((err = _fdt_splice (fdt, p, 0, newlen))) | ||||||
|  |     return err; | ||||||
|  | 
 | ||||||
|  |   fdt_set_size_dt_strings (fdt, fdt_size_dt_strings (fdt) + newlen); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _fdt_find_add_string (void *fdt, const char *s) | ||||||
|  | { | ||||||
|  |   char *strtab = (char *) fdt + fdt_off_dt_strings (fdt); | ||||||
|  |   const char *p; | ||||||
|  |   char *new; | ||||||
|  |   int len = strlen (s) + 1; | ||||||
|  |   int err; | ||||||
|  | 
 | ||||||
|  |   p = _fdt_find_string (strtab, fdt_size_dt_strings (fdt), s); | ||||||
|  |   if (p) | ||||||
|  |     /* found it */ | ||||||
|  |     return (p - strtab); | ||||||
|  | 
 | ||||||
|  |   new = strtab + fdt_size_dt_strings (fdt); | ||||||
|  |   err = _fdt_splice_string (fdt, len); | ||||||
|  |   if (err) | ||||||
|  |     return err; | ||||||
|  | 
 | ||||||
|  |   memcpy (new, s, len); | ||||||
|  |   return (new - strtab); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_add_mem_rsv (void *fdt, uint64_t address, uint64_t size) | ||||||
|  | { | ||||||
|  |   struct fdt_reserve_entry *re; | ||||||
|  |   int err; | ||||||
|  | 
 | ||||||
|  |   FDT_RW_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   re = _fdt_mem_rsv_w (fdt, fdt_num_mem_rsv (fdt)); | ||||||
|  |   err = _fdt_splice_mem_rsv (fdt, re, 0, 1); | ||||||
|  |   if (err) | ||||||
|  |     return err; | ||||||
|  | 
 | ||||||
|  |   re->address = cpu_to_fdt64 (address); | ||||||
|  |   re->size = cpu_to_fdt64 (size); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_del_mem_rsv (void *fdt, int n) | ||||||
|  | { | ||||||
|  |   struct fdt_reserve_entry *re = _fdt_mem_rsv_w (fdt, n); | ||||||
|  |   int err; | ||||||
|  | 
 | ||||||
|  |   FDT_RW_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   if (n >= fdt_num_mem_rsv (fdt)) | ||||||
|  |     return -FDT_ERR_NOTFOUND; | ||||||
|  | 
 | ||||||
|  |   err = _fdt_splice_mem_rsv (fdt, re, 1, 0); | ||||||
|  |   if (err) | ||||||
|  |     return err; | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _fdt_resize_property (void *fdt, int nodeoffset, const char *name, | ||||||
|  | 		      int len, struct fdt_property **prop) | ||||||
|  | { | ||||||
|  |   int oldlen; | ||||||
|  |   int err; | ||||||
|  | 
 | ||||||
|  |   *prop = fdt_get_property_w (fdt, nodeoffset, name, &oldlen); | ||||||
|  |   if (!(*prop)) | ||||||
|  |     return oldlen; | ||||||
|  | 
 | ||||||
|  |   if ((err = _fdt_splice_struct (fdt, (*prop)->data, FDT_TAGALIGN (oldlen), | ||||||
|  | 				 FDT_TAGALIGN (len)))) | ||||||
|  |     return err; | ||||||
|  | 
 | ||||||
|  |   (*prop)->len = cpu_to_fdt32 (len); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _fdt_add_property (void *fdt, int nodeoffset, const char *name, | ||||||
|  | 		   int len, struct fdt_property **prop) | ||||||
|  | { | ||||||
|  |   int proplen; | ||||||
|  |   int nextoffset; | ||||||
|  |   int namestroff; | ||||||
|  |   int err; | ||||||
|  | 
 | ||||||
|  |   if ((nextoffset = _fdt_check_node_offset (fdt, nodeoffset)) < 0) | ||||||
|  |     return nextoffset; | ||||||
|  | 
 | ||||||
|  |   namestroff = _fdt_find_add_string (fdt, name); | ||||||
|  |   if (namestroff < 0) | ||||||
|  |     return namestroff; | ||||||
|  | 
 | ||||||
|  |   *prop = _fdt_offset_ptr_w (fdt, nextoffset); | ||||||
|  |   proplen = sizeof (**prop) + FDT_TAGALIGN (len); | ||||||
|  | 
 | ||||||
|  |   err = _fdt_splice_struct (fdt, *prop, 0, proplen); | ||||||
|  |   if (err) | ||||||
|  |     return err; | ||||||
|  | 
 | ||||||
|  |   (*prop)->tag = cpu_to_fdt32 (FDT_PROP); | ||||||
|  |   (*prop)->nameoff = cpu_to_fdt32 (namestroff); | ||||||
|  |   (*prop)->len = cpu_to_fdt32 (len); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_set_name (void *fdt, int nodeoffset, const char *name) | ||||||
|  | { | ||||||
|  |   char *namep; | ||||||
|  |   int oldlen, newlen; | ||||||
|  |   int err; | ||||||
|  | 
 | ||||||
|  |   FDT_RW_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   namep = (char *) (uintptr_t) fdt_get_name (fdt, nodeoffset, &oldlen); | ||||||
|  |   if (!namep) | ||||||
|  |     return oldlen; | ||||||
|  | 
 | ||||||
|  |   newlen = strlen (name); | ||||||
|  | 
 | ||||||
|  |   err = _fdt_splice_struct (fdt, namep, FDT_TAGALIGN (oldlen + 1), | ||||||
|  | 			    FDT_TAGALIGN (newlen + 1)); | ||||||
|  |   if (err) | ||||||
|  |     return err; | ||||||
|  | 
 | ||||||
|  |   memcpy (namep, name, newlen + 1); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_setprop (void *fdt, int nodeoffset, const char *name, | ||||||
|  | 	     const void *val, int len) | ||||||
|  | { | ||||||
|  |   struct fdt_property *prop; | ||||||
|  |   int err; | ||||||
|  | 
 | ||||||
|  |   FDT_RW_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   err = _fdt_resize_property (fdt, nodeoffset, name, len, &prop); | ||||||
|  |   if (err == -FDT_ERR_NOTFOUND) | ||||||
|  |     err = _fdt_add_property (fdt, nodeoffset, name, len, &prop); | ||||||
|  |   if (err) | ||||||
|  |     return err; | ||||||
|  | 
 | ||||||
|  |   memcpy (prop->data, val, len); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_delprop (void *fdt, int nodeoffset, const char *name) | ||||||
|  | { | ||||||
|  |   struct fdt_property *prop; | ||||||
|  |   int len, proplen; | ||||||
|  | 
 | ||||||
|  |   FDT_RW_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   prop = fdt_get_property_w (fdt, nodeoffset, name, &len); | ||||||
|  |   if (!prop) | ||||||
|  |     return len; | ||||||
|  | 
 | ||||||
|  |   proplen = sizeof (*prop) + FDT_TAGALIGN (len); | ||||||
|  |   return _fdt_splice_struct (fdt, prop, proplen, 0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_add_subnode_namelen (void *fdt, int parentoffset, | ||||||
|  | 			 const char *name, int namelen) | ||||||
|  | { | ||||||
|  |   struct fdt_node_header *nh; | ||||||
|  |   int offset, nextoffset; | ||||||
|  |   int nodelen; | ||||||
|  |   int err; | ||||||
|  |   uint32_t tag; | ||||||
|  |   uint32_t *endtag; | ||||||
|  | 
 | ||||||
|  |   FDT_RW_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   offset = fdt_subnode_offset_namelen (fdt, parentoffset, name, namelen); | ||||||
|  |   if (offset >= 0) | ||||||
|  |     return -FDT_ERR_EXISTS; | ||||||
|  |   else if (offset != -FDT_ERR_NOTFOUND) | ||||||
|  |     return offset; | ||||||
|  | 
 | ||||||
|  |   /* Try to place the new node after the parent's properties */ | ||||||
|  |   fdt_next_tag (fdt, parentoffset, &nextoffset);	/* skip the BEGIN_NODE */ | ||||||
|  |   do | ||||||
|  |     { | ||||||
|  |       offset = nextoffset; | ||||||
|  |       tag = fdt_next_tag (fdt, offset, &nextoffset); | ||||||
|  |     } | ||||||
|  |   while ((tag == FDT_PROP) || (tag == FDT_NOP)); | ||||||
|  | 
 | ||||||
|  |   nh = _fdt_offset_ptr_w (fdt, offset); | ||||||
|  |   nodelen = sizeof (*nh) + FDT_TAGALIGN (namelen + 1) + FDT_TAGSIZE; | ||||||
|  | 
 | ||||||
|  |   err = _fdt_splice_struct (fdt, nh, 0, nodelen); | ||||||
|  |   if (err) | ||||||
|  |     return err; | ||||||
|  | 
 | ||||||
|  |   nh->tag = cpu_to_fdt32 (FDT_BEGIN_NODE); | ||||||
|  |   memset (nh->name, 0, FDT_TAGALIGN (namelen + 1)); | ||||||
|  |   memcpy (nh->name, name, namelen); | ||||||
|  |   endtag = (uint32_t *) ((char *) nh + nodelen - FDT_TAGSIZE); | ||||||
|  |   *endtag = cpu_to_fdt32 (FDT_END_NODE); | ||||||
|  | 
 | ||||||
|  |   return offset; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_add_subnode (void *fdt, int parentoffset, const char *name) | ||||||
|  | { | ||||||
|  |   return fdt_add_subnode_namelen (fdt, parentoffset, name, strlen (name)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_del_node (void *fdt, int nodeoffset) | ||||||
|  | { | ||||||
|  |   int endoffset; | ||||||
|  | 
 | ||||||
|  |   FDT_RW_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   endoffset = _fdt_node_end_offset (fdt, nodeoffset); | ||||||
|  |   if (endoffset < 0) | ||||||
|  |     return endoffset; | ||||||
|  | 
 | ||||||
|  |   return _fdt_splice_struct (fdt, _fdt_offset_ptr_w (fdt, nodeoffset), | ||||||
|  | 			     endoffset - nodeoffset, 0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void | ||||||
|  | _fdt_packblocks (const char *old, char *new, | ||||||
|  | 		 int mem_rsv_size, int struct_size) | ||||||
|  | { | ||||||
|  |   int mem_rsv_off, struct_off, strings_off; | ||||||
|  | 
 | ||||||
|  |   mem_rsv_off = FDT_ALIGN (sizeof (struct fdt_header), 8); | ||||||
|  |   struct_off = mem_rsv_off + mem_rsv_size; | ||||||
|  |   strings_off = struct_off + struct_size; | ||||||
|  | 
 | ||||||
|  |   memmove (new + mem_rsv_off, old + fdt_off_mem_rsvmap (old), mem_rsv_size); | ||||||
|  |   fdt_set_off_mem_rsvmap (new, mem_rsv_off); | ||||||
|  | 
 | ||||||
|  |   memmove (new + struct_off, old + fdt_off_dt_struct (old), struct_size); | ||||||
|  |   fdt_set_off_dt_struct (new, struct_off); | ||||||
|  |   fdt_set_size_dt_struct (new, struct_size); | ||||||
|  | 
 | ||||||
|  |   memmove (new + strings_off, old + fdt_off_dt_strings (old), | ||||||
|  | 	   fdt_size_dt_strings (old)); | ||||||
|  |   fdt_set_off_dt_strings (new, strings_off); | ||||||
|  |   fdt_set_size_dt_strings (new, fdt_size_dt_strings (old)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_open_into (const void *fdt, void *buf, int bufsize) | ||||||
|  | { | ||||||
|  |   int err; | ||||||
|  |   int mem_rsv_size, struct_size; | ||||||
|  |   int newsize; | ||||||
|  |   const char *fdtstart = fdt; | ||||||
|  |   const char *fdtend = fdtstart + fdt_totalsize (fdt); | ||||||
|  |   char *tmp; | ||||||
|  | 
 | ||||||
|  |   FDT_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   mem_rsv_size = (fdt_num_mem_rsv (fdt) + 1) | ||||||
|  |     * sizeof (struct fdt_reserve_entry); | ||||||
|  | 
 | ||||||
|  |   if (fdt_version (fdt) >= 17) | ||||||
|  |     { | ||||||
|  |       struct_size = fdt_size_dt_struct (fdt); | ||||||
|  |     } | ||||||
|  |   else | ||||||
|  |     { | ||||||
|  |       struct_size = 0; | ||||||
|  |       while (fdt_next_tag (fdt, struct_size, &struct_size) != FDT_END) | ||||||
|  | 	; | ||||||
|  |       if (struct_size < 0) | ||||||
|  | 	return struct_size; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   if (!_fdt_blocks_misordered (fdt, mem_rsv_size, struct_size)) | ||||||
|  |     { | ||||||
|  |       /* no further work necessary */ | ||||||
|  |       err = fdt_move (fdt, buf, bufsize); | ||||||
|  |       if (err) | ||||||
|  | 	return err; | ||||||
|  |       fdt_set_version (buf, 17); | ||||||
|  |       fdt_set_size_dt_struct (buf, struct_size); | ||||||
|  |       fdt_set_totalsize (buf, bufsize); | ||||||
|  |       return 0; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   /* Need to reorder */ | ||||||
|  |   newsize = FDT_ALIGN (sizeof (struct fdt_header), 8) + mem_rsv_size | ||||||
|  |     + struct_size + fdt_size_dt_strings (fdt); | ||||||
|  | 
 | ||||||
|  |   if (bufsize < newsize) | ||||||
|  |     return -FDT_ERR_NOSPACE; | ||||||
|  | 
 | ||||||
|  |   /* First attempt to build converted tree at beginning of buffer */ | ||||||
|  |   tmp = buf; | ||||||
|  |   /* But if that overlaps with the old tree... */ | ||||||
|  |   if (((tmp + newsize) > fdtstart) && (tmp < fdtend)) | ||||||
|  |     { | ||||||
|  |       /* Try right after the old tree instead */ | ||||||
|  |       tmp = (char *) (uintptr_t) fdtend; | ||||||
|  |       if ((tmp + newsize) > ((char *) buf + bufsize)) | ||||||
|  | 	return -FDT_ERR_NOSPACE; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   _fdt_packblocks (fdt, tmp, mem_rsv_size, struct_size); | ||||||
|  |   memmove (buf, tmp, newsize); | ||||||
|  | 
 | ||||||
|  |   fdt_set_magic (buf, FDT_MAGIC); | ||||||
|  |   fdt_set_totalsize (buf, bufsize); | ||||||
|  |   fdt_set_version (buf, 17); | ||||||
|  |   fdt_set_last_comp_version (buf, 16); | ||||||
|  |   fdt_set_boot_cpuid_phys (buf, fdt_boot_cpuid_phys (fdt)); | ||||||
|  | 
 | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_pack (void *fdt) | ||||||
|  | { | ||||||
|  |   int mem_rsv_size; | ||||||
|  | 
 | ||||||
|  |   FDT_RW_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   mem_rsv_size = (fdt_num_mem_rsv (fdt) + 1) | ||||||
|  |     * sizeof (struct fdt_reserve_entry); | ||||||
|  |   _fdt_packblocks (fdt, fdt, mem_rsv_size, fdt_size_dt_struct (fdt)); | ||||||
|  |   fdt_set_totalsize (fdt, _fdt_data_size (fdt)); | ||||||
|  | 
 | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
							
								
								
									
										100
									
								
								grub-core/lib/dtc/libfdt/fdt_strerror.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								grub-core/lib/dtc/libfdt/fdt_strerror.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,100 @@ | ||||||
|  | /*
 | ||||||
|  |  * libfdt - Flat Device Tree manipulation | ||||||
|  |  * Copyright (C) 2006 David Gibson, IBM Corporation. | ||||||
|  |  * | ||||||
|  |  * libfdt is dual licensed: you can use it either under the terms of | ||||||
|  |  * the GPL, or the BSD license, at your option. | ||||||
|  |  * | ||||||
|  |  *  a) This library 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 2 of the | ||||||
|  |  *     License, or (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  *     This library 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 this library; if not, write to the Free | ||||||
|  |  *     Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, | ||||||
|  |  *     MA 02110-1301 USA | ||||||
|  |  * | ||||||
|  |  * Alternatively, | ||||||
|  |  * | ||||||
|  |  *  b) 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. | ||||||
|  |  * | ||||||
|  |  *     THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS 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 COPYRIGHT OWNER 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. | ||||||
|  |  */ | ||||||
|  | #include "libfdt_env.h" | ||||||
|  | 
 | ||||||
|  | #include <fdt.h> | ||||||
|  | #include <libfdt.h> | ||||||
|  | 
 | ||||||
|  | #include "libfdt_internal.h" | ||||||
|  | 
 | ||||||
|  | struct fdt_errtabent | ||||||
|  | { | ||||||
|  |   const char *str; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define FDT_ERRTABENT(val) \ | ||||||
|  | 	[(val)] = { .str = #val, } | ||||||
|  | 
 | ||||||
|  | static struct fdt_errtabent fdt_errtable[] = { | ||||||
|  |   FDT_ERRTABENT (FDT_ERR_NOTFOUND), | ||||||
|  |   FDT_ERRTABENT (FDT_ERR_EXISTS), | ||||||
|  |   FDT_ERRTABENT (FDT_ERR_NOSPACE), | ||||||
|  | 
 | ||||||
|  |   FDT_ERRTABENT (FDT_ERR_BADOFFSET), | ||||||
|  |   FDT_ERRTABENT (FDT_ERR_BADPATH), | ||||||
|  |   FDT_ERRTABENT (FDT_ERR_BADSTATE), | ||||||
|  | 
 | ||||||
|  |   FDT_ERRTABENT (FDT_ERR_TRUNCATED), | ||||||
|  |   FDT_ERRTABENT (FDT_ERR_BADMAGIC), | ||||||
|  |   FDT_ERRTABENT (FDT_ERR_BADVERSION), | ||||||
|  |   FDT_ERRTABENT (FDT_ERR_BADSTRUCTURE), | ||||||
|  |   FDT_ERRTABENT (FDT_ERR_BADLAYOUT), | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define FDT_ERRTABSIZE	(sizeof(fdt_errtable) / sizeof(fdt_errtable[0])) | ||||||
|  | 
 | ||||||
|  | const char * | ||||||
|  | fdt_strerror (int errval) | ||||||
|  | { | ||||||
|  |   if (errval > 0) | ||||||
|  |     return "<valid offset/length>"; | ||||||
|  |   else if (errval == 0) | ||||||
|  |     return "<no error>"; | ||||||
|  |   else if (errval > -FDT_ERRTABSIZE) | ||||||
|  |     { | ||||||
|  |       const char *s = fdt_errtable[-errval].str; | ||||||
|  | 
 | ||||||
|  |       if (s) | ||||||
|  | 	return s; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   return "<unknown error>"; | ||||||
|  | } | ||||||
							
								
								
									
										267
									
								
								grub-core/lib/dtc/libfdt/fdt_sw.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										267
									
								
								grub-core/lib/dtc/libfdt/fdt_sw.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,267 @@ | ||||||
|  | /*
 | ||||||
|  |  * libfdt - Flat Device Tree manipulation | ||||||
|  |  * Copyright (C) 2006 David Gibson, IBM Corporation. | ||||||
|  |  * | ||||||
|  |  * libfdt is dual licensed: you can use it either under the terms of | ||||||
|  |  * the GPL, or the BSD license, at your option. | ||||||
|  |  * | ||||||
|  |  *  a) This library 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 2 of the | ||||||
|  |  *     License, or (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  *     This library 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 this library; if not, write to the Free | ||||||
|  |  *     Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, | ||||||
|  |  *     MA 02110-1301 USA | ||||||
|  |  * | ||||||
|  |  * Alternatively, | ||||||
|  |  * | ||||||
|  |  *  b) 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. | ||||||
|  |  * | ||||||
|  |  *     THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS 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 COPYRIGHT OWNER 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. | ||||||
|  |  */ | ||||||
|  | #include "libfdt_env.h" | ||||||
|  | 
 | ||||||
|  | #include <fdt.h> | ||||||
|  | #include <libfdt.h> | ||||||
|  | 
 | ||||||
|  | #include "libfdt_internal.h" | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _fdt_sw_check_header (void *fdt) | ||||||
|  | { | ||||||
|  |   if (fdt_magic (fdt) != FDT_SW_MAGIC) | ||||||
|  |     return -FDT_ERR_BADMAGIC; | ||||||
|  |   /* FIXME: should check more details about the header state */ | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #define FDT_SW_CHECK_HEADER(fdt) \ | ||||||
|  | 	{ \ | ||||||
|  | 		int err; \ | ||||||
|  | 		if ((err = _fdt_sw_check_header(fdt)) != 0) \ | ||||||
|  | 			return err; \ | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | static void * | ||||||
|  | _fdt_grab_space (void *fdt, size_t len) | ||||||
|  | { | ||||||
|  |   int offset = fdt_size_dt_struct (fdt); | ||||||
|  |   int spaceleft; | ||||||
|  | 
 | ||||||
|  |   spaceleft = fdt_totalsize (fdt) - fdt_off_dt_struct (fdt) | ||||||
|  |     - fdt_size_dt_strings (fdt); | ||||||
|  | 
 | ||||||
|  |   if ((offset + len < offset) || (offset + len > spaceleft)) | ||||||
|  |     return NULL; | ||||||
|  | 
 | ||||||
|  |   fdt_set_size_dt_struct (fdt, offset + len); | ||||||
|  |   return _fdt_offset_ptr_w (fdt, offset); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_create (void *buf, int bufsize) | ||||||
|  | { | ||||||
|  |   void *fdt = buf; | ||||||
|  | 
 | ||||||
|  |   if (bufsize < sizeof (struct fdt_header)) | ||||||
|  |     return -FDT_ERR_NOSPACE; | ||||||
|  | 
 | ||||||
|  |   memset (buf, 0, bufsize); | ||||||
|  | 
 | ||||||
|  |   fdt_set_magic (fdt, FDT_SW_MAGIC); | ||||||
|  |   fdt_set_version (fdt, FDT_LAST_SUPPORTED_VERSION); | ||||||
|  |   fdt_set_last_comp_version (fdt, FDT_FIRST_SUPPORTED_VERSION); | ||||||
|  |   fdt_set_totalsize (fdt, bufsize); | ||||||
|  | 
 | ||||||
|  |   fdt_set_off_mem_rsvmap (fdt, FDT_ALIGN (sizeof (struct fdt_header), | ||||||
|  | 					  sizeof (struct fdt_reserve_entry))); | ||||||
|  |   fdt_set_off_dt_struct (fdt, fdt_off_mem_rsvmap (fdt)); | ||||||
|  |   fdt_set_off_dt_strings (fdt, bufsize); | ||||||
|  | 
 | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_add_reservemap_entry (void *fdt, uint64_t addr, uint64_t size) | ||||||
|  | { | ||||||
|  |   struct fdt_reserve_entry *re; | ||||||
|  |   int offset; | ||||||
|  | 
 | ||||||
|  |   FDT_SW_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   if (fdt_size_dt_struct (fdt)) | ||||||
|  |     return -FDT_ERR_BADSTATE; | ||||||
|  | 
 | ||||||
|  |   offset = fdt_off_dt_struct (fdt); | ||||||
|  |   if ((offset + sizeof (*re)) > fdt_totalsize (fdt)) | ||||||
|  |     return -FDT_ERR_NOSPACE; | ||||||
|  | 
 | ||||||
|  |   re = (struct fdt_reserve_entry *) ((char *) fdt + offset); | ||||||
|  |   re->address = cpu_to_fdt64 (addr); | ||||||
|  |   re->size = cpu_to_fdt64 (size); | ||||||
|  | 
 | ||||||
|  |   fdt_set_off_dt_struct (fdt, offset + sizeof (*re)); | ||||||
|  | 
 | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_finish_reservemap (void *fdt) | ||||||
|  | { | ||||||
|  |   return fdt_add_reservemap_entry (fdt, 0, 0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_begin_node (void *fdt, const char *name) | ||||||
|  | { | ||||||
|  |   struct fdt_node_header *nh; | ||||||
|  |   int namelen = strlen (name) + 1; | ||||||
|  | 
 | ||||||
|  |   FDT_SW_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   nh = _fdt_grab_space (fdt, sizeof (*nh) + FDT_TAGALIGN (namelen)); | ||||||
|  |   if (!nh) | ||||||
|  |     return -FDT_ERR_NOSPACE; | ||||||
|  | 
 | ||||||
|  |   nh->tag = cpu_to_fdt32 (FDT_BEGIN_NODE); | ||||||
|  |   memcpy (nh->name, name, namelen); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_end_node (void *fdt) | ||||||
|  | { | ||||||
|  |   uint32_t *en; | ||||||
|  | 
 | ||||||
|  |   FDT_SW_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   en = _fdt_grab_space (fdt, FDT_TAGSIZE); | ||||||
|  |   if (!en) | ||||||
|  |     return -FDT_ERR_NOSPACE; | ||||||
|  | 
 | ||||||
|  |   *en = cpu_to_fdt32 (FDT_END_NODE); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | _fdt_find_add_string (void *fdt, const char *s) | ||||||
|  | { | ||||||
|  |   char *strtab = (char *) fdt + fdt_totalsize (fdt); | ||||||
|  |   const char *p; | ||||||
|  |   int strtabsize = fdt_size_dt_strings (fdt); | ||||||
|  |   int len = strlen (s) + 1; | ||||||
|  |   int struct_top, offset; | ||||||
|  | 
 | ||||||
|  |   p = _fdt_find_string (strtab - strtabsize, strtabsize, s); | ||||||
|  |   if (p) | ||||||
|  |     return p - strtab; | ||||||
|  | 
 | ||||||
|  |   /* Add it */ | ||||||
|  |   offset = -strtabsize - len; | ||||||
|  |   struct_top = fdt_off_dt_struct (fdt) + fdt_size_dt_struct (fdt); | ||||||
|  |   if (fdt_totalsize (fdt) + offset < struct_top) | ||||||
|  |     return 0;			/* no more room  */ | ||||||
|  | 
 | ||||||
|  |   memcpy (strtab + offset, s, len); | ||||||
|  |   fdt_set_size_dt_strings (fdt, strtabsize + len); | ||||||
|  |   return offset; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_property (void *fdt, const char *name, const void *val, int len) | ||||||
|  | { | ||||||
|  |   struct fdt_property *prop; | ||||||
|  |   int nameoff; | ||||||
|  | 
 | ||||||
|  |   FDT_SW_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   nameoff = _fdt_find_add_string (fdt, name); | ||||||
|  |   if (nameoff == 0) | ||||||
|  |     return -FDT_ERR_NOSPACE; | ||||||
|  | 
 | ||||||
|  |   prop = _fdt_grab_space (fdt, sizeof (*prop) + FDT_TAGALIGN (len)); | ||||||
|  |   if (!prop) | ||||||
|  |     return -FDT_ERR_NOSPACE; | ||||||
|  | 
 | ||||||
|  |   prop->tag = cpu_to_fdt32 (FDT_PROP); | ||||||
|  |   prop->nameoff = cpu_to_fdt32 (nameoff); | ||||||
|  |   prop->len = cpu_to_fdt32 (len); | ||||||
|  |   memcpy (prop->data, val, len); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_finish (void *fdt) | ||||||
|  | { | ||||||
|  |   char *p = (char *) fdt; | ||||||
|  |   uint32_t *end; | ||||||
|  |   int oldstroffset, newstroffset; | ||||||
|  |   uint32_t tag; | ||||||
|  |   int offset, nextoffset; | ||||||
|  | 
 | ||||||
|  |   FDT_SW_CHECK_HEADER (fdt); | ||||||
|  | 
 | ||||||
|  |   /* Add terminator */ | ||||||
|  |   end = _fdt_grab_space (fdt, sizeof (*end)); | ||||||
|  |   if (!end) | ||||||
|  |     return -FDT_ERR_NOSPACE; | ||||||
|  |   *end = cpu_to_fdt32 (FDT_END); | ||||||
|  | 
 | ||||||
|  |   /* Relocate the string table */ | ||||||
|  |   oldstroffset = fdt_totalsize (fdt) - fdt_size_dt_strings (fdt); | ||||||
|  |   newstroffset = fdt_off_dt_struct (fdt) + fdt_size_dt_struct (fdt); | ||||||
|  |   memmove (p + newstroffset, p + oldstroffset, fdt_size_dt_strings (fdt)); | ||||||
|  |   fdt_set_off_dt_strings (fdt, newstroffset); | ||||||
|  | 
 | ||||||
|  |   /* Walk the structure, correcting string offsets */ | ||||||
|  |   offset = 0; | ||||||
|  |   while ((tag = fdt_next_tag (fdt, offset, &nextoffset)) != FDT_END) | ||||||
|  |     { | ||||||
|  |       if (tag == FDT_PROP) | ||||||
|  | 	{ | ||||||
|  | 	  struct fdt_property *prop = _fdt_offset_ptr_w (fdt, offset); | ||||||
|  | 	  int nameoff; | ||||||
|  | 
 | ||||||
|  | 	  nameoff = fdt32_to_cpu (prop->nameoff); | ||||||
|  | 	  nameoff += fdt_size_dt_strings (fdt); | ||||||
|  | 	  prop->nameoff = cpu_to_fdt32 (nameoff); | ||||||
|  | 	} | ||||||
|  |       offset = nextoffset; | ||||||
|  |     } | ||||||
|  |   if (nextoffset < 0) | ||||||
|  |     return nextoffset; | ||||||
|  | 
 | ||||||
|  |   /* Finally, adjust the header */ | ||||||
|  |   fdt_set_totalsize (fdt, newstroffset + fdt_size_dt_strings (fdt)); | ||||||
|  |   fdt_set_magic (fdt, FDT_MAGIC); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
							
								
								
									
										123
									
								
								grub-core/lib/dtc/libfdt/fdt_wip.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										123
									
								
								grub-core/lib/dtc/libfdt/fdt_wip.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,123 @@ | ||||||
|  | /*
 | ||||||
|  |  * libfdt - Flat Device Tree manipulation | ||||||
|  |  * Copyright (C) 2006 David Gibson, IBM Corporation. | ||||||
|  |  * | ||||||
|  |  * libfdt is dual licensed: you can use it either under the terms of | ||||||
|  |  * the GPL, or the BSD license, at your option. | ||||||
|  |  * | ||||||
|  |  *  a) This library 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 2 of the | ||||||
|  |  *     License, or (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  *     This library 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 this library; if not, write to the Free | ||||||
|  |  *     Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, | ||||||
|  |  *     MA 02110-1301 USA | ||||||
|  |  * | ||||||
|  |  * Alternatively, | ||||||
|  |  * | ||||||
|  |  *  b) 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. | ||||||
|  |  * | ||||||
|  |  *     THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS 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 COPYRIGHT OWNER 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. | ||||||
|  |  */ | ||||||
|  | #include "libfdt_env.h" | ||||||
|  | 
 | ||||||
|  | #include <fdt.h> | ||||||
|  | #include <libfdt.h> | ||||||
|  | 
 | ||||||
|  | #include "libfdt_internal.h" | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_setprop_inplace (void *fdt, int nodeoffset, const char *name, | ||||||
|  | 		     const void *val, int len) | ||||||
|  | { | ||||||
|  |   void *propval; | ||||||
|  |   int proplen; | ||||||
|  | 
 | ||||||
|  |   propval = fdt_getprop_w (fdt, nodeoffset, name, &proplen); | ||||||
|  |   if (!propval) | ||||||
|  |     return proplen; | ||||||
|  | 
 | ||||||
|  |   if (proplen != len) | ||||||
|  |     return -FDT_ERR_NOSPACE; | ||||||
|  | 
 | ||||||
|  |   memcpy (propval, val, len); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void | ||||||
|  | _fdt_nop_region (void *start, int len) | ||||||
|  | { | ||||||
|  |   uint32_t *p; | ||||||
|  | 
 | ||||||
|  |   for (p = start; (char *) p < ((char *) start + len); p++) | ||||||
|  |     *p = cpu_to_fdt32 (FDT_NOP); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_nop_property (void *fdt, int nodeoffset, const char *name) | ||||||
|  | { | ||||||
|  |   struct fdt_property *prop; | ||||||
|  |   int len; | ||||||
|  | 
 | ||||||
|  |   prop = fdt_get_property_w (fdt, nodeoffset, name, &len); | ||||||
|  |   if (!prop) | ||||||
|  |     return len; | ||||||
|  | 
 | ||||||
|  |   _fdt_nop_region (prop, len + sizeof (*prop)); | ||||||
|  | 
 | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | _fdt_node_end_offset (void *fdt, int offset) | ||||||
|  | { | ||||||
|  |   int depth = 0; | ||||||
|  | 
 | ||||||
|  |   while ((offset >= 0) && (depth >= 0)) | ||||||
|  |     offset = fdt_next_node (fdt, offset, &depth); | ||||||
|  | 
 | ||||||
|  |   return offset; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | fdt_nop_node (void *fdt, int nodeoffset) | ||||||
|  | { | ||||||
|  |   int endoffset; | ||||||
|  | 
 | ||||||
|  |   endoffset = _fdt_node_end_offset (fdt, nodeoffset); | ||||||
|  |   if (endoffset < 0) | ||||||
|  |     return endoffset; | ||||||
|  | 
 | ||||||
|  |   _fdt_nop_region (fdt_offset_ptr_w (fdt, nodeoffset, 0), | ||||||
|  | 		   endoffset - nodeoffset); | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
							
								
								
									
										1239
									
								
								grub-core/lib/dtc/libfdt/libfdt.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1239
									
								
								grub-core/lib/dtc/libfdt/libfdt.h
									
										
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
							
								
								
									
										27
									
								
								grub-core/lib/dtc/libfdt/libfdt_env.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								grub-core/lib/dtc/libfdt/libfdt_env.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | ||||||
|  | #ifndef _LIBFDT_ENV_H | ||||||
|  | #define _LIBFDT_ENV_H | ||||||
|  | 
 | ||||||
|  | #include <stddef.h> | ||||||
|  | #include <stdint.h> | ||||||
|  | #include <string.h> | ||||||
|  | 
 | ||||||
|  | #define _B(n)	((unsigned long long)((uint8_t *)&x)[n]) | ||||||
|  | static inline uint32_t | ||||||
|  | fdt32_to_cpu (uint32_t x) | ||||||
|  | { | ||||||
|  |   return (_B (0) << 24) | (_B (1) << 16) | (_B (2) << 8) | _B (3); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #define cpu_to_fdt32(x) fdt32_to_cpu(x) | ||||||
|  | 
 | ||||||
|  | static inline uint64_t | ||||||
|  | fdt64_to_cpu (uint64_t x) | ||||||
|  | { | ||||||
|  |   return (_B (0) << 56) | (_B (1) << 48) | (_B (2) << 40) | (_B (3) << 32) | ||||||
|  |     | (_B (4) << 24) | (_B (5) << 16) | (_B (6) << 8) | _B (7); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #define cpu_to_fdt64(x) fdt64_to_cpu(x) | ||||||
|  | #undef _B | ||||||
|  | 
 | ||||||
|  | #endif /* _LIBFDT_ENV_H */ | ||||||
							
								
								
									
										100
									
								
								grub-core/lib/dtc/libfdt/libfdt_internal.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								grub-core/lib/dtc/libfdt/libfdt_internal.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,100 @@ | ||||||
|  | #ifndef _LIBFDT_INTERNAL_H | ||||||
|  | #define _LIBFDT_INTERNAL_H | ||||||
|  | /*
 | ||||||
|  |  * libfdt - Flat Device Tree manipulation | ||||||
|  |  * Copyright (C) 2006 David Gibson, IBM Corporation. | ||||||
|  |  * | ||||||
|  |  * libfdt is dual licensed: you can use it either under the terms of | ||||||
|  |  * the GPL, or the BSD license, at your option. | ||||||
|  |  * | ||||||
|  |  *  a) This library 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 2 of the | ||||||
|  |  *     License, or (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  *     This library 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 this library; if not, write to the Free | ||||||
|  |  *     Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, | ||||||
|  |  *     MA 02110-1301 USA | ||||||
|  |  * | ||||||
|  |  * Alternatively, | ||||||
|  |  * | ||||||
|  |  *  b) 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. | ||||||
|  |  * | ||||||
|  |  *     THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS 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 COPYRIGHT OWNER 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. | ||||||
|  |  */ | ||||||
|  | #include <fdt.h> | ||||||
|  | 
 | ||||||
|  | #define FDT_ALIGN(x, a)		(((x) + (a) - 1) & ~((a) - 1)) | ||||||
|  | #define FDT_TAGALIGN(x)		(FDT_ALIGN((x), FDT_TAGSIZE)) | ||||||
|  | 
 | ||||||
|  | #define FDT_CHECK_HEADER(fdt) \ | ||||||
|  | 	{ \ | ||||||
|  | 		int err; \ | ||||||
|  | 		if ((err = fdt_check_header(fdt)) != 0) \ | ||||||
|  | 			return err; \ | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | int _fdt_check_node_offset (const void *fdt, int offset); | ||||||
|  | int _fdt_check_prop_offset (const void *fdt, int offset); | ||||||
|  | const char *_fdt_find_string (const char *strtab, int tabsize, const char *s); | ||||||
|  | int _fdt_node_end_offset (void *fdt, int nodeoffset); | ||||||
|  | 
 | ||||||
|  | static inline const void * | ||||||
|  | _fdt_offset_ptr (const void *fdt, int offset) | ||||||
|  | { | ||||||
|  |   return (const char *) fdt + fdt_off_dt_struct (fdt) + offset; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void * | ||||||
|  | _fdt_offset_ptr_w (void *fdt, int offset) | ||||||
|  | { | ||||||
|  |   return (void *) (uintptr_t) _fdt_offset_ptr (fdt, offset); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline const struct fdt_reserve_entry * | ||||||
|  | _fdt_mem_rsv (const void *fdt, int n) | ||||||
|  | { | ||||||
|  |   const struct fdt_reserve_entry *rsv_table = | ||||||
|  |     (const struct fdt_reserve_entry *) | ||||||
|  |     ((const char *) fdt + fdt_off_mem_rsvmap (fdt)); | ||||||
|  | 
 | ||||||
|  |   return rsv_table + n; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline struct fdt_reserve_entry * | ||||||
|  | _fdt_mem_rsv_w (void *fdt, int n) | ||||||
|  | { | ||||||
|  |   return (void *) (uintptr_t) _fdt_mem_rsv (fdt, n); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #define FDT_SW_MAGIC		(~FDT_MAGIC) | ||||||
|  | 
 | ||||||
|  | #endif /* _LIBFDT_INTERNAL_H */ | ||||||
							
								
								
									
										54
									
								
								grub-core/lib/dtc/libfdt/version.lds
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								grub-core/lib/dtc/libfdt/version.lds
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,54 @@ | ||||||
|  | LIBFDT_1.2 { | ||||||
|  | 	global: | ||||||
|  | 		fdt_next_node; | ||||||
|  | 		fdt_check_header; | ||||||
|  | 		fdt_move; | ||||||
|  | 		fdt_string; | ||||||
|  | 		fdt_num_mem_rsv; | ||||||
|  | 		fdt_get_mem_rsv; | ||||||
|  | 		fdt_subnode_offset_namelen; | ||||||
|  | 		fdt_subnode_offset; | ||||||
|  | 		fdt_path_offset; | ||||||
|  | 		fdt_get_name; | ||||||
|  | 		fdt_get_property_namelen; | ||||||
|  | 		fdt_get_property; | ||||||
|  | 		fdt_getprop_namelen; | ||||||
|  | 		fdt_getprop; | ||||||
|  | 		fdt_get_phandle; | ||||||
|  | 		fdt_get_alias_namelen; | ||||||
|  | 		fdt_get_alias; | ||||||
|  | 		fdt_get_path; | ||||||
|  | 		fdt_supernode_atdepth_offset; | ||||||
|  | 		fdt_node_depth; | ||||||
|  | 		fdt_parent_offset; | ||||||
|  | 		fdt_node_offset_by_prop_value; | ||||||
|  | 		fdt_node_offset_by_phandle; | ||||||
|  | 		fdt_node_check_compatible; | ||||||
|  | 		fdt_node_offset_by_compatible; | ||||||
|  | 		fdt_setprop_inplace; | ||||||
|  | 		fdt_nop_property; | ||||||
|  | 		fdt_nop_node; | ||||||
|  | 		fdt_create; | ||||||
|  | 		fdt_add_reservemap_entry; | ||||||
|  | 		fdt_finish_reservemap; | ||||||
|  | 		fdt_begin_node; | ||||||
|  | 		fdt_property; | ||||||
|  | 		fdt_end_node; | ||||||
|  | 		fdt_finish; | ||||||
|  | 		fdt_open_into; | ||||||
|  | 		fdt_pack; | ||||||
|  | 		fdt_add_mem_rsv; | ||||||
|  | 		fdt_del_mem_rsv; | ||||||
|  | 		fdt_set_name; | ||||||
|  | 		fdt_setprop; | ||||||
|  | 		fdt_delprop; | ||||||
|  | 		fdt_add_subnode_namelen; | ||||||
|  | 		fdt_add_subnode; | ||||||
|  | 		fdt_del_node; | ||||||
|  | 		fdt_strerror; | ||||||
|  | 		fdt_offset_ptr; | ||||||
|  | 		fdt_next_tag; | ||||||
|  | 
 | ||||||
|  | 	local: | ||||||
|  | 		*; | ||||||
|  | }; | ||||||
|  | @ -28,7 +28,7 @@ void | ||||||
| grub_halt (void) | grub_halt (void) | ||||||
| { | { | ||||||
|   grub_machine_fini (); |   grub_machine_fini (); | ||||||
| #ifndef __ia64__ | #if !defined(__ia64__) && !defined(__arm__) | ||||||
|   grub_acpi_halt (); |   grub_acpi_halt (); | ||||||
| #endif | #endif | ||||||
|   efi_call_4 (grub_efi_system_table->runtime_services->reset_system, |   efi_call_4 (grub_efi_system_table->runtime_services->reset_system, | ||||||
|  |  | ||||||
|  | @ -11,6 +11,8 @@ | ||||||
| #elif defined(__ia64__) | #elif defined(__ia64__) | ||||||
| #include "./ia64/setjmp.S" | #include "./ia64/setjmp.S" | ||||||
| #include "./ia64/longjmp.S" | #include "./ia64/longjmp.S" | ||||||
|  | #elif defined(__arm__) | ||||||
|  | #include "./arm/setjmp.S" | ||||||
| #else | #else | ||||||
| #error "Unknown target cpu type" | #error "Unknown target cpu type" | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
							
								
								
									
										41
									
								
								grub-core/lib/uboot/datetime.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								grub-core/lib/uboot/datetime.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,41 @@ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/types.h> | ||||||
|  | #include <grub/symbol.h> | ||||||
|  | #include <grub/uboot/uboot.h> | ||||||
|  | #include <grub/datetime.h> | ||||||
|  | #include <grub/dl.h> | ||||||
|  | 
 | ||||||
|  | GRUB_MOD_LICENSE ("GPLv3+"); | ||||||
|  | 
 | ||||||
|  | /* No simple platform-independent RTC access exists in U-Boot. */ | ||||||
|  | 
 | ||||||
|  | grub_err_t | ||||||
|  | grub_get_datetime (struct grub_datetime *datetime __attribute__ ((unused))) | ||||||
|  | { | ||||||
|  |   return grub_error (GRUB_ERR_INVALID_COMMAND, | ||||||
|  | 		     "can\'t get datetime using U-Boot"); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | grub_err_t | ||||||
|  | grub_set_datetime (struct grub_datetime * datetime __attribute__ ((unused))) | ||||||
|  | { | ||||||
|  |   return grub_error (GRUB_ERR_INVALID_COMMAND, | ||||||
|  | 		     "can\'t set datetime using U-Boot"); | ||||||
|  | } | ||||||
							
								
								
									
										31
									
								
								grub-core/lib/uboot/halt.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								grub-core/lib/uboot/halt.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,31 @@ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/misc.h> | ||||||
|  | #include <grub/mm.h> | ||||||
|  | #include <grub/kernel.h> | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | grub_halt (void) | ||||||
|  | { | ||||||
|  |   grub_machine_fini (); | ||||||
|  | 
 | ||||||
|  |   /* Just stop here */ | ||||||
|  | 
 | ||||||
|  |   while (1); | ||||||
|  | } | ||||||
							
								
								
									
										30
									
								
								grub-core/lib/uboot/reboot.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								grub-core/lib/uboot/reboot.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,30 @@ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/kernel.h> | ||||||
|  | #include <grub/misc.h> | ||||||
|  | #include <grub/uboot/uboot.h> | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | grub_reboot (void) | ||||||
|  | { | ||||||
|  |   grub_machine_fini (); | ||||||
|  | 
 | ||||||
|  |   uboot_reset (); | ||||||
|  |   while (1); | ||||||
|  | } | ||||||
							
								
								
									
										405
									
								
								grub-core/loader/arm/linux.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										405
									
								
								grub-core/loader/arm/linux.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,405 @@ | ||||||
|  | /* linux.c - boot Linux */ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/dl.h> | ||||||
|  | #include <grub/file.h> | ||||||
|  | #include <grub/loader.h> | ||||||
|  | #include <grub/mm.h> | ||||||
|  | #include <grub/misc.h> | ||||||
|  | #include <grub/command.h> | ||||||
|  | #include <grub/cache.h> | ||||||
|  | #include <grub/cpu/linux.h> | ||||||
|  | #include <grub/lib/cmdline.h> | ||||||
|  | 
 | ||||||
|  | #include <libfdt.h> | ||||||
|  | 
 | ||||||
|  | GRUB_MOD_LICENSE ("GPLv3+"); | ||||||
|  | 
 | ||||||
|  | static grub_dl_t my_mod; | ||||||
|  | 
 | ||||||
|  | static grub_addr_t initrd_start; | ||||||
|  | static grub_size_t initrd_end; | ||||||
|  | 
 | ||||||
|  | static grub_addr_t linux_addr; | ||||||
|  | static grub_size_t linux_size; | ||||||
|  | 
 | ||||||
|  | static char *linux_args; | ||||||
|  | 
 | ||||||
|  | static grub_addr_t firmware_boot_data; | ||||||
|  | static grub_addr_t boot_data; | ||||||
|  | static grub_uint32_t machine_type; | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * linux_prepare_fdt(): | ||||||
|  |  *   Prepares a loaded FDT for being passed to Linux. | ||||||
|  |  *   Merges in command line parameters and sets up initrd addresses. | ||||||
|  |  */ | ||||||
|  | static grub_err_t | ||||||
|  | linux_prepare_fdt (void) | ||||||
|  | { | ||||||
|  |   int node; | ||||||
|  |   int retval; | ||||||
|  |   int tmp_size; | ||||||
|  |   void *tmp_fdt; | ||||||
|  | 
 | ||||||
|  |   tmp_size = fdt_totalsize ((void *) boot_data) + FDT_ADDITIONAL_ENTRIES_SIZE; | ||||||
|  |   tmp_fdt = grub_malloc (tmp_size); | ||||||
|  |   if (!tmp_fdt) | ||||||
|  |     return GRUB_ERR_OUT_OF_MEMORY; | ||||||
|  | 
 | ||||||
|  |   fdt_open_into ((void *) boot_data, tmp_fdt, tmp_size); | ||||||
|  | 
 | ||||||
|  |   /* Find or create '/chosen' node */ | ||||||
|  |   node = fdt_subnode_offset (tmp_fdt, 0, "chosen"); | ||||||
|  |   if (node < 0) | ||||||
|  |     { | ||||||
|  |       grub_printf ("No 'chosen' node in FDT - creating.\n"); | ||||||
|  |       node = fdt_add_subnode (tmp_fdt, 0, "chosen"); | ||||||
|  |       if (node < 0) | ||||||
|  | 	goto failure; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   grub_printf ("linux_args: '%s'\n", linux_args); | ||||||
|  | 
 | ||||||
|  |   /* Generate and set command line */ | ||||||
|  |   retval = fdt_setprop (tmp_fdt, node, "bootargs", linux_args, | ||||||
|  | 			grub_strlen (linux_args) + 1); | ||||||
|  |   if (retval) | ||||||
|  |     goto failure; | ||||||
|  | 
 | ||||||
|  |   if (initrd_start && initrd_end) | ||||||
|  |     { | ||||||
|  |       /*
 | ||||||
|  |        * We're using physical addresses, so even if we have LPAE, we're | ||||||
|  |        * restricted to a 32-bit address space. | ||||||
|  |        */ | ||||||
|  |       grub_uint32_t fdt_initrd_start = cpu_to_fdt32 (initrd_start); | ||||||
|  |       grub_uint32_t fdt_initrd_end = cpu_to_fdt32 (initrd_end); | ||||||
|  | 
 | ||||||
|  |       grub_dprintf ("loader", "Initrd @ 0x%08x-0x%08x\n", | ||||||
|  | 		    initrd_start, initrd_end); | ||||||
|  | 
 | ||||||
|  |       retval = fdt_setprop (tmp_fdt, node, "linux,initrd-start", | ||||||
|  | 			    &fdt_initrd_start, sizeof (fdt_initrd_start)); | ||||||
|  |       if (retval) | ||||||
|  | 	goto failure; | ||||||
|  |       retval = fdt_setprop (tmp_fdt, node, "linux,initrd-end", | ||||||
|  | 			    &fdt_initrd_end, sizeof (fdt_initrd_end)); | ||||||
|  |       if (retval) | ||||||
|  | 	goto failure; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   /* Copy updated FDT to its launch location */ | ||||||
|  |   fdt_move (tmp_fdt, (void *) boot_data, fdt_totalsize (tmp_fdt)); | ||||||
|  |   grub_free (tmp_fdt); | ||||||
|  |   fdt_pack ((void *) boot_data); | ||||||
|  | 
 | ||||||
|  |   grub_dprintf ("loader", "FDT updated for Linux boot\n"); | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | 
 | ||||||
|  | failure: | ||||||
|  |   grub_free (tmp_fdt); | ||||||
|  |   return GRUB_ERR_BAD_ARGUMENT; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static grub_err_t | ||||||
|  | linux_boot (void) | ||||||
|  | { | ||||||
|  |   kernel_entry_t linuxmain; | ||||||
|  |   grub_err_t err = GRUB_ERR_NONE; | ||||||
|  | 
 | ||||||
|  |   grub_arch_sync_caches ((void *) linux_addr, linux_size); | ||||||
|  | 
 | ||||||
|  |   grub_dprintf ("loader", "Kernel at: 0x%x\n", linux_addr); | ||||||
|  | 
 | ||||||
|  |   if (!boot_data) | ||||||
|  |     { | ||||||
|  |       if (firmware_boot_data) | ||||||
|  | 	{ | ||||||
|  | 	  grub_printf ("Using firmware-supplied boot data @ 0x%08x\n", | ||||||
|  | 		       firmware_boot_data); | ||||||
|  | 	  boot_data = firmware_boot_data; | ||||||
|  | 	} | ||||||
|  |       else | ||||||
|  | 	{ | ||||||
|  | 	  return GRUB_ERR_FILE_NOT_FOUND; | ||||||
|  | 	} | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   grub_dprintf ("loader", "Boot data at: 0x%x\n", boot_data); | ||||||
|  | 
 | ||||||
|  |   if (fdt32_to_cpu (*(grub_uint32_t *) (boot_data)) == FDT_MAGIC) | ||||||
|  |     { | ||||||
|  |       grub_dprintf ("loader", "FDT @ 0x%08x\n", (grub_addr_t) boot_data); | ||||||
|  |       if (linux_prepare_fdt () != GRUB_ERR_NONE) | ||||||
|  | 	{ | ||||||
|  | 	  grub_dprintf ("loader", "linux_prepare_fdt() failed\n"); | ||||||
|  | 	  return GRUB_ERR_FILE_NOT_FOUND; | ||||||
|  | 	} | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   grub_dprintf ("loader", "Jumping to Linux...\n"); | ||||||
|  | 
 | ||||||
|  |   /* Boot the kernel.
 | ||||||
|  |    *   Arguments to kernel: | ||||||
|  |    *     r0 - 0 | ||||||
|  |    *     r1 - machine type (possibly passed from firmware) | ||||||
|  |    *     r2 - address of DTB or ATAG list | ||||||
|  |    */ | ||||||
|  |   linuxmain = (kernel_entry_t) linux_addr; | ||||||
|  | 
 | ||||||
|  | #ifdef GRUB_MACHINE_EFI | ||||||
|  |   err = grub_efi_prepare_platform(); | ||||||
|  |   if (err != GRUB_ERR_NONE) | ||||||
|  |     return err; | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  |   linuxmain (0, machine_type, (void *) boot_data); | ||||||
|  | 
 | ||||||
|  |   return err; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Only support zImage, so no relocations necessary | ||||||
|  |  */ | ||||||
|  | static grub_err_t | ||||||
|  | linux_load (const char *filename) | ||||||
|  | { | ||||||
|  |   grub_file_t file; | ||||||
|  |   int size; | ||||||
|  | 
 | ||||||
|  |   file = grub_file_open (filename); | ||||||
|  |   if (!file) | ||||||
|  |     return GRUB_ERR_FILE_NOT_FOUND; | ||||||
|  | 
 | ||||||
|  |   size = grub_file_size (file); | ||||||
|  |   if (size == 0) | ||||||
|  |     return GRUB_ERR_FILE_READ_ERROR; | ||||||
|  | 
 | ||||||
|  | #ifdef GRUB_MACHINE_EFI | ||||||
|  |   linux_addr = (grub_addr_t) grub_efi_allocate_loader_memory (LINUX_PHYS_OFFSET, size); | ||||||
|  | #else | ||||||
|  |   linux_addr = LINUX_ADDRESS; | ||||||
|  | #endif | ||||||
|  |   grub_dprintf ("loader", "Loading Linux to 0x%08x\n", | ||||||
|  | 		(grub_addr_t) linux_addr); | ||||||
|  | 
 | ||||||
|  |   if (grub_file_read (file, (void *) linux_addr, size) != size) | ||||||
|  |     { | ||||||
|  |       grub_printf ("Kernel read failed!\n"); | ||||||
|  |       return GRUB_ERR_FILE_READ_ERROR; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   if (*(grub_uint32_t *) (linux_addr + LINUX_ZIMAGE_OFFSET) | ||||||
|  |       != LINUX_ZIMAGE_MAGIC) | ||||||
|  |     { | ||||||
|  |       return grub_error (GRUB_ERR_BAD_FILE_TYPE, N_("invalid zImage")); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   linux_size = size; | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static grub_err_t | ||||||
|  | linux_unload (void) | ||||||
|  | { | ||||||
|  |   grub_dl_unref (my_mod); | ||||||
|  | 
 | ||||||
|  |   grub_free (linux_args); | ||||||
|  |   linux_args = NULL; | ||||||
|  | 
 | ||||||
|  |   initrd_start = initrd_end = 0; | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static grub_err_t | ||||||
|  | grub_cmd_linux (grub_command_t cmd __attribute__ ((unused)), | ||||||
|  | 		int argc, char *argv[]) | ||||||
|  | { | ||||||
|  |   int size, retval; | ||||||
|  |   grub_file_t file; | ||||||
|  |   grub_dl_ref (my_mod); | ||||||
|  | 
 | ||||||
|  |   if (argc == 0) | ||||||
|  |     return grub_error (GRUB_ERR_BAD_ARGUMENT, N_("filename expected")); | ||||||
|  | 
 | ||||||
|  |   file = grub_file_open (argv[0]); | ||||||
|  |   if (!file) | ||||||
|  |     goto fail; | ||||||
|  | 
 | ||||||
|  |   retval = linux_load (argv[0]); | ||||||
|  |   grub_file_close (file); | ||||||
|  |   if (retval != GRUB_ERR_NONE) | ||||||
|  |     goto fail; | ||||||
|  | 
 | ||||||
|  |   grub_loader_set (linux_boot, linux_unload, 1); | ||||||
|  | 
 | ||||||
|  |   size = grub_loader_cmdline_size (argc, argv); | ||||||
|  |   linux_args = grub_malloc (size + sizeof (LINUX_IMAGE)); | ||||||
|  |   if (!linux_args) | ||||||
|  |     goto fail; | ||||||
|  | 
 | ||||||
|  |   /* Create kernel command line.  */ | ||||||
|  |   grub_memcpy (linux_args, LINUX_IMAGE, sizeof (LINUX_IMAGE)); | ||||||
|  |   grub_create_loader_cmdline (argc, argv, | ||||||
|  | 			      linux_args + sizeof (LINUX_IMAGE) - 1, size); | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | 
 | ||||||
|  | fail: | ||||||
|  |   grub_dl_unref (my_mod); | ||||||
|  |   return grub_errno; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static grub_err_t | ||||||
|  | grub_cmd_initrd (grub_command_t cmd __attribute__ ((unused)), | ||||||
|  | 		 int argc, char *argv[]) | ||||||
|  | { | ||||||
|  |   grub_file_t file; | ||||||
|  |   int size; | ||||||
|  | 
 | ||||||
|  |   if (argc == 0) | ||||||
|  |     return grub_error (GRUB_ERR_BAD_ARGUMENT, N_("filename expected")); | ||||||
|  | 
 | ||||||
|  |   file = grub_file_open (argv[0]); | ||||||
|  |   if (!file) | ||||||
|  |     return grub_errno; | ||||||
|  | 
 | ||||||
|  |   size = grub_file_size (file); | ||||||
|  |   if (size == 0) | ||||||
|  |     goto fail; | ||||||
|  | 
 | ||||||
|  | #ifdef GRUB_MACHINE_EFI | ||||||
|  |   initrd_start = (grub_addr_t) grub_efi_allocate_loader_memory (LINUX_INITRD_PHYS_OFFSET, size); | ||||||
|  | #else | ||||||
|  |   initrd_start = LINUX_INITRD_ADDRESS; | ||||||
|  | #endif | ||||||
|  |   grub_dprintf ("loader", "Loading initrd to 0x%08x\n", | ||||||
|  | 		(grub_addr_t) initrd_start); | ||||||
|  | 
 | ||||||
|  |   if (grub_file_read (file, (void *) initrd_start, size) != size) | ||||||
|  |     goto fail; | ||||||
|  | 
 | ||||||
|  |   initrd_end = initrd_start + size; | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | 
 | ||||||
|  | fail: | ||||||
|  |   grub_file_close (file); | ||||||
|  | 
 | ||||||
|  |   return grub_errno; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void * | ||||||
|  | load_dtb (grub_file_t dtb, int size) | ||||||
|  | { | ||||||
|  |   void *fdt; | ||||||
|  | 
 | ||||||
|  |   fdt = grub_malloc (size); | ||||||
|  |   if (!fdt) | ||||||
|  |     return NULL; | ||||||
|  | 
 | ||||||
|  |   if (grub_file_read (dtb, fdt, size) != size) | ||||||
|  |     { | ||||||
|  |       grub_free (fdt); | ||||||
|  |       return NULL; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   if (fdt_check_header (fdt) != 0) | ||||||
|  |     { | ||||||
|  |       grub_free (fdt); | ||||||
|  |       return NULL; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |   return fdt; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static grub_err_t | ||||||
|  | grub_cmd_devicetree (grub_command_t cmd __attribute__ ((unused)), | ||||||
|  | 		     int argc, char *argv[]) | ||||||
|  | { | ||||||
|  |   grub_file_t dtb; | ||||||
|  |   void *blob; | ||||||
|  |   int size; | ||||||
|  | 
 | ||||||
|  |   if (argc != 1) | ||||||
|  |     return grub_error (GRUB_ERR_BAD_ARGUMENT, N_("filename expected")); | ||||||
|  | 
 | ||||||
|  |   dtb = grub_file_open (argv[0]); | ||||||
|  |   if (!dtb) | ||||||
|  |     return grub_error (GRUB_ERR_FILE_NOT_FOUND, N_("failed to open file")); | ||||||
|  | 
 | ||||||
|  |   size = grub_file_size (dtb); | ||||||
|  |   if (size == 0) | ||||||
|  |     goto out; | ||||||
|  | 
 | ||||||
|  |   blob = load_dtb (dtb, size); | ||||||
|  |   if (!blob) | ||||||
|  |     return GRUB_ERR_FILE_NOT_FOUND; | ||||||
|  | 
 | ||||||
|  | #ifdef GRUB_MACHINE_EFI | ||||||
|  |   boot_data = (grub_addr_t) grub_efi_allocate_loader_memory (LINUX_FDT_PHYS_OFFSET, size); | ||||||
|  | #else | ||||||
|  |   boot_data = LINUX_FDT_ADDRESS; | ||||||
|  | #endif | ||||||
|  |   grub_dprintf ("loader", "Loading device tree to 0x%08x\n", | ||||||
|  | 		(grub_addr_t) boot_data); | ||||||
|  |   fdt_move (blob, (void *) boot_data, fdt_totalsize (blob)); | ||||||
|  |   grub_free (blob); | ||||||
|  | 
 | ||||||
|  |   /* 
 | ||||||
|  |    * We've successfully loaded an FDT, so any machine type passed | ||||||
|  |    * from firmware is now obsolete. | ||||||
|  |    */ | ||||||
|  |   machine_type = ARM_FDT_MACHINE_TYPE; | ||||||
|  | 
 | ||||||
|  |   return GRUB_ERR_NONE; | ||||||
|  | 
 | ||||||
|  |  out: | ||||||
|  |   grub_file_close (dtb); | ||||||
|  | 
 | ||||||
|  |   return grub_errno; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static grub_command_t cmd_linux, cmd_initrd, cmd_devicetree; | ||||||
|  | 
 | ||||||
|  | GRUB_MOD_INIT (linux) | ||||||
|  | { | ||||||
|  |   cmd_linux = grub_register_command ("linux", grub_cmd_linux, | ||||||
|  | 				     0, N_("Load Linux.")); | ||||||
|  |   cmd_initrd = grub_register_command ("initrd", grub_cmd_initrd, | ||||||
|  | 				      0, N_("Load initrd.")); | ||||||
|  |   cmd_devicetree = grub_register_command ("devicetree", grub_cmd_devicetree, | ||||||
|  | 					  0, N_("Load DTB file.")); | ||||||
|  |   my_mod = mod; | ||||||
|  |   firmware_boot_data = firmware_get_boot_data (); | ||||||
|  | 
 | ||||||
|  |   boot_data = (grub_addr_t) NULL; | ||||||
|  |   machine_type = firmware_get_machine_type (); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | GRUB_MOD_FINI (linux) | ||||||
|  | { | ||||||
|  |   grub_unregister_command (cmd_linux); | ||||||
|  |   grub_unregister_command (cmd_initrd); | ||||||
|  |   grub_unregister_command (cmd_devicetree); | ||||||
|  | } | ||||||
|  | @ -745,7 +745,9 @@ grub_cmd_terminfo (grub_extcmd_context_t ctxt, int argc, char **args) | ||||||
| 
 | 
 | ||||||
| static grub_extcmd_t cmd; | static grub_extcmd_t cmd; | ||||||
| 
 | 
 | ||||||
| #if defined (GRUB_MACHINE_IEEE1275) || defined (GRUB_MACHINE_MIPS_LOONGSON) || defined (GRUB_MACHINE_MIPS_QEMU_MIPS) || defined (GRUB_MACHINE_ARC) | #if defined (GRUB_MACHINE_IEEE1275) || defined (GRUB_MACHINE_MIPS_LOONGSON) \ | ||||||
|  |   || defined (GRUB_MACHINE_MIPS_QEMU_MIPS) || defined (GRUB_MACHINE_ARC) \ | ||||||
|  |   || defined (GRUB_MACHINE_UBOOT) | ||||||
| void grub_terminfo_init (void) | void grub_terminfo_init (void) | ||||||
| #else | #else | ||||||
| GRUB_MOD_INIT(terminfo) | GRUB_MOD_INIT(terminfo) | ||||||
|  |  | ||||||
							
								
								
									
										141
									
								
								grub-core/term/uboot/console.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										141
									
								
								grub-core/term/uboot/console.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,141 @@ | ||||||
|  | /* console.c - console interface layer for U-Boot platforms */ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/term.h> | ||||||
|  | #include <grub/misc.h> | ||||||
|  | #include <grub/types.h> | ||||||
|  | #include <grub/err.h> | ||||||
|  | #include <grub/terminfo.h> | ||||||
|  | #include <grub/uboot/uboot.h> | ||||||
|  | #include <grub/uboot/console.h> | ||||||
|  | 
 | ||||||
|  | static void | ||||||
|  | put (struct grub_term_output *term __attribute__ ((unused)), const int c) | ||||||
|  | { | ||||||
|  |   uboot_putc (c); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | readkey (struct grub_term_input *term __attribute__ ((unused))) | ||||||
|  | { | ||||||
|  |   if (uboot_tstc () > 0) | ||||||
|  |     return uboot_getc (); | ||||||
|  | 
 | ||||||
|  |   return -1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void | ||||||
|  | uboot_console_setcursor (struct grub_term_output *term | ||||||
|  | 			 __attribute__ ((unused)), int on | ||||||
|  | 			 __attribute__ ((unused))) | ||||||
|  | { | ||||||
|  |   grub_terminfo_setcursor (term, on); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static grub_err_t | ||||||
|  | uboot_console_init_input (struct grub_term_input *term) | ||||||
|  | { | ||||||
|  |   return grub_terminfo_input_init (term); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | extern struct grub_terminfo_output_state uboot_console_terminfo_output; | ||||||
|  | 
 | ||||||
|  | static void | ||||||
|  | uboot_console_dimensions (void) | ||||||
|  | { | ||||||
|  |   /* Use a small console by default.  */ | ||||||
|  |   if (!uboot_console_terminfo_output.width) | ||||||
|  |     uboot_console_terminfo_output.width = 80; | ||||||
|  |   if (!uboot_console_terminfo_output.height) | ||||||
|  |     uboot_console_terminfo_output.height = 24; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static grub_err_t | ||||||
|  | uboot_console_init_output (struct grub_term_output *term) | ||||||
|  | { | ||||||
|  |   uboot_console_dimensions (); | ||||||
|  | 
 | ||||||
|  |   grub_terminfo_output_init (term); | ||||||
|  | 
 | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct grub_terminfo_input_state uboot_console_terminfo_input = { | ||||||
|  |   .readkey = readkey | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct grub_terminfo_output_state uboot_console_terminfo_output = { | ||||||
|  |   .put = put, | ||||||
|  |   .width = 80, | ||||||
|  |   .height = 24 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static struct grub_term_input uboot_console_term_input = { | ||||||
|  |   .name = "console", | ||||||
|  |   .init = uboot_console_init_input, | ||||||
|  |   .getkey = grub_terminfo_getkey, | ||||||
|  |   .data = &uboot_console_terminfo_input | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static struct grub_term_output uboot_console_term_output = { | ||||||
|  |   .name = "console", | ||||||
|  |   .init = uboot_console_init_output, | ||||||
|  |   .putchar = grub_terminfo_putchar, | ||||||
|  |   .getwh = grub_terminfo_getwh, | ||||||
|  |   .getxy = grub_terminfo_getxy, | ||||||
|  |   .gotoxy = grub_terminfo_gotoxy, | ||||||
|  |   .cls = grub_terminfo_cls, | ||||||
|  |   .setcolorstate = grub_terminfo_setcolorstate, | ||||||
|  |   .setcursor = uboot_console_setcursor, | ||||||
|  |   .flags = GRUB_TERM_CODE_TYPE_ASCII, | ||||||
|  |   .data = &uboot_console_terminfo_output, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | grub_console_init_early (void) | ||||||
|  | { | ||||||
|  |   grub_term_register_input ("console", &uboot_console_term_input); | ||||||
|  |   grub_term_register_output ("console", &uboot_console_term_output); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * grub_console_init_lately(): | ||||||
|  |  *   Initializes terminfo formatting by registering terminal type. | ||||||
|  |  *   Called after heap has been configured. | ||||||
|  |  *    | ||||||
|  |  */ | ||||||
|  | void | ||||||
|  | grub_console_init_lately (void) | ||||||
|  | { | ||||||
|  |   const char *type; | ||||||
|  | 
 | ||||||
|  |   /* See if explicitly set by U-Boot environment */ | ||||||
|  |   type = uboot_env_get ("grub_term"); | ||||||
|  |   if (!type) | ||||||
|  |     type = "vt100"; | ||||||
|  | 
 | ||||||
|  |   grub_terminfo_init (); | ||||||
|  |   grub_terminfo_output_register (&uboot_console_term_output, type); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | grub_console_fini (void) | ||||||
|  | { | ||||||
|  | } | ||||||
							
								
								
									
										26
									
								
								include/grub/arm/efi/loader.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								include/grub/arm/efi/loader.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,26 @@ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/>.
 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef GRUB_LOADER_MACHINE_HEADER | ||||||
|  | #define GRUB_LOADER_MACHINE_HEADER	1 | ||||||
|  | 
 | ||||||
|  | grub_err_t EXPORT_FUNC (grub_efi_prepare_platform) (void); | ||||||
|  | void * EXPORT_FUNC (grub_efi_allocate_loader_memory) (grub_uint32_t min_offset, | ||||||
|  | 						      grub_uint32_t size); | ||||||
|  | 
 | ||||||
|  | #endif /* ! GRUB_LOADER_MACHINE_HEADER */ | ||||||
							
								
								
									
										1
									
								
								include/grub/arm/efi/memory.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								include/grub/arm/efi/memory.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1 @@ | ||||||
|  | #include <grub/efi/memory.h> | ||||||
							
								
								
									
										59
									
								
								include/grub/arm/linux.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								include/grub/arm/linux.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,59 @@ | ||||||
|  | /* linux.h - ARM linux specific definitions */ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/>.
 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef GRUB_LINUX_CPU_HEADER | ||||||
|  | #define GRUB_LINUX_CPU_HEADER 1 | ||||||
|  | 
 | ||||||
|  | #define LINUX_ZIMAGE_OFFSET 0x24 | ||||||
|  | #define LINUX_ZIMAGE_MAGIC  0x016f2818 | ||||||
|  | 
 | ||||||
|  | #define ARM_FDT_MACHINE_TYPE 0xFFFFFFFF | ||||||
|  | 
 | ||||||
|  | #if defined GRUB_MACHINE_UBOOT | ||||||
|  | # include <grub/uboot/uboot.h> | ||||||
|  | # define LINUX_ADDRESS        (start_of_ram + 0x8000) | ||||||
|  | # define LINUX_INITRD_ADDRESS (start_of_ram + 0x02000000) | ||||||
|  | # define LINUX_FDT_ADDRESS    (LINUX_INITRD_ADDRESS - 0x10000) | ||||||
|  | # define firmware_get_boot_data uboot_get_boot_data | ||||||
|  | # define firmware_get_machine_type uboot_get_machine_type | ||||||
|  | #elif defined GRUB_MACHINE_EFI | ||||||
|  | # include <grub/efi/efi.h> | ||||||
|  | # include <grub/machine/loader.h> | ||||||
|  | /* On UEFI platforms - load the images at the lowest available address not
 | ||||||
|  |    less than *_PHYS_OFFSET from the first available memory location. */ | ||||||
|  | # define LINUX_PHYS_OFFSET        (0x00008000) | ||||||
|  | # define LINUX_INITRD_PHYS_OFFSET (LINUX_PHYS_OFFSET + 0x02000000) | ||||||
|  | # define LINUX_FDT_PHYS_OFFSET    (LINUX_INITRD_PHYS_OFFSET - 0x10000) | ||||||
|  | static inline grub_addr_t | ||||||
|  | firmware_get_boot_data (void) | ||||||
|  | { | ||||||
|  |   return 0; | ||||||
|  | } | ||||||
|  | static inline grub_uint32_t | ||||||
|  | firmware_get_machine_type (void) | ||||||
|  | { | ||||||
|  |   return ARM_FDT_MACHINE_TYPE; | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #define FDT_ADDITIONAL_ENTRIES_SIZE	0x300 | ||||||
|  | 
 | ||||||
|  | typedef void (*kernel_entry_t) (int, unsigned long, void *); | ||||||
|  | 
 | ||||||
|  | #endif /* ! GRUB_LINUX_CPU_HEADER */ | ||||||
							
								
								
									
										7
									
								
								include/grub/arm/system.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								include/grub/arm/system.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,7 @@ | ||||||
|  | #ifndef GRUB_SYSTEM_CPU_HEADER | ||||||
|  | #define GRUB_SYSTEM_CPU_HEADER | ||||||
|  | 
 | ||||||
|  | void grub_arm_disable_caches_mmu (void); | ||||||
|  | 
 | ||||||
|  | #endif /* ! GRUB_SYSTEM_CPU_HEADER */ | ||||||
|  | 
 | ||||||
							
								
								
									
										29
									
								
								include/grub/arm/time.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								include/grub/arm/time.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,29 @@ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/>.
 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef KERNEL_CPU_TIME_HEADER | ||||||
|  | #define KERNEL_CPU_TIME_HEADER	1 | ||||||
|  | 
 | ||||||
|  | static __inline void | ||||||
|  | grub_cpu_idle (void) | ||||||
|  | { | ||||||
|  |   /* FIXME: this can't work until we handle interrupts.  */ | ||||||
|  | /*  __asm__ __volatile__ ("wfi"); */ | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #endif /* ! KERNEL_CPU_TIME_HEADER */ | ||||||
							
								
								
									
										34
									
								
								include/grub/arm/types.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								include/grub/arm/types.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,34 @@ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/>.
 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef GRUB_TYPES_CPU_HEADER | ||||||
|  | #define GRUB_TYPES_CPU_HEADER	1 | ||||||
|  | 
 | ||||||
|  | /* The size of void *.  */ | ||||||
|  | #define GRUB_TARGET_SIZEOF_VOID_P	4 | ||||||
|  | 
 | ||||||
|  | /* The size of long.  */ | ||||||
|  | #define GRUB_TARGET_SIZEOF_LONG		4 | ||||||
|  | 
 | ||||||
|  | /* currently only support little-endian.  */ | ||||||
|  | #undef GRUB_TARGET_WORDS_BIGENDIAN | ||||||
|  | 
 | ||||||
|  | /* Unaligned accesses only supported if MMU enabled */ | ||||||
|  | //#define GRUB_HAVE_UNALIGNED_ACCESS 1
 | ||||||
|  | 
 | ||||||
|  | #endif /* ! GRUB_TYPES_CPU_HEADER */ | ||||||
							
								
								
									
										32
									
								
								include/grub/arm/uboot/kernel.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								include/grub/arm/uboot/kernel.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,32 @@ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013 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/>.
 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef GRUB_KERNEL_MACHINE_HEADER | ||||||
|  | #define GRUB_KERNEL_MACHINE_HEADER	1 | ||||||
|  | 
 | ||||||
|  | #ifndef ASM_FILE | ||||||
|  | 
 | ||||||
|  | #include <grub/symbol.h> | ||||||
|  | #include <grub/types.h> | ||||||
|  | 
 | ||||||
|  | #endif /* ! ASM_FILE */ | ||||||
|  | 
 | ||||||
|  | #define GRUB_KERNEL_MACHINE_STACK_SIZE 0x40000 | ||||||
|  | #define GRUB_KERNEL_MACHINE_HEAP_SIZE  (grub_size_t) (2 * 1024 * 1024) | ||||||
|  | 
 | ||||||
|  | #endif /* ! GRUB_KERNEL_MACHINE_HEADER */ | ||||||
|  | @ -44,6 +44,7 @@ enum grub_disk_dev_id | ||||||
|     GRUB_DISK_DEVICE_ARCDISK_ID, |     GRUB_DISK_DEVICE_ARCDISK_ID, | ||||||
|     GRUB_DISK_DEVICE_HOSTDISK_ID, |     GRUB_DISK_DEVICE_HOSTDISK_ID, | ||||||
|     GRUB_DISK_DEVICE_PROCFS_ID, |     GRUB_DISK_DEVICE_PROCFS_ID, | ||||||
|  |     GRUB_DISK_DEVICE_UBOOTDISK_ID, | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
| struct grub_disk; | struct grub_disk; | ||||||
|  |  | ||||||
|  | @ -63,9 +63,10 @@ struct grub_pe32_coff_header | ||||||
|   grub_uint16_t characteristics; |   grub_uint16_t characteristics; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| #define GRUB_PE32_MACHINE_I386		0x14c | #define GRUB_PE32_MACHINE_I386			0x14c | ||||||
| #define GRUB_PE32_MACHINE_IA64		0x200 | #define GRUB_PE32_MACHINE_IA64			0x200 | ||||||
| #define GRUB_PE32_MACHINE_X86_64	0x8664 | #define GRUB_PE32_MACHINE_X86_64		0x8664 | ||||||
|  | #define GRUB_PE32_MACHINE_ARMTHUMB_MIXED	0x01c2 | ||||||
| 
 | 
 | ||||||
| #define GRUB_PE32_RELOCS_STRIPPED		0x0001 | #define GRUB_PE32_RELOCS_STRIPPED		0x0001 | ||||||
| #define GRUB_PE32_EXECUTABLE_IMAGE		0x0002 | #define GRUB_PE32_EXECUTABLE_IMAGE		0x0002 | ||||||
|  | @ -276,8 +277,10 @@ struct grub_pe32_fixup_block | ||||||
| #define GRUB_PE32_REL_BASED_HIGHLOW	3 | #define GRUB_PE32_REL_BASED_HIGHLOW	3 | ||||||
| #define GRUB_PE32_REL_BASED_HIGHADJ	4 | #define GRUB_PE32_REL_BASED_HIGHADJ	4 | ||||||
| #define GRUB_PE32_REL_BASED_MIPS_JMPADDR 5 | #define GRUB_PE32_REL_BASED_MIPS_JMPADDR 5 | ||||||
|  | #define GRUB_PE32_REL_BASED_ARM_MOV32A  5 | ||||||
| #define GRUB_PE32_REL_BASED_SECTION	6 | #define GRUB_PE32_REL_BASED_SECTION	6 | ||||||
| #define GRUB_PE32_REL_BASED_REL		7 | #define GRUB_PE32_REL_BASED_REL		7 | ||||||
|  | #define GRUB_PE32_REL_BASED_ARM_MOV32T  7 | ||||||
| #define GRUB_PE32_REL_BASED_IA64_IMM64	9 | #define GRUB_PE32_REL_BASED_IA64_IMM64	9 | ||||||
| #define GRUB_PE32_REL_BASED_DIR64	10 | #define GRUB_PE32_REL_BASED_DIR64	10 | ||||||
| #define GRUB_PE32_REL_BASED_HIGH3ADJ	11 | #define GRUB_PE32_REL_BASED_HIGH3ADJ	11 | ||||||
|  |  | ||||||
|  | @ -78,7 +78,7 @@ struct grub_module_info64 | ||||||
| #if defined (GRUB_MACHINE_PCBIOS) || defined (GRUB_MACHINE_COREBOOT) \ | #if defined (GRUB_MACHINE_PCBIOS) || defined (GRUB_MACHINE_COREBOOT) \ | ||||||
|   || defined (GRUB_MACHINE_MULTIBOOT) || defined (GRUB_MACHINE_MIPS_QEMU_MIPS) \ |   || defined (GRUB_MACHINE_MULTIBOOT) || defined (GRUB_MACHINE_MIPS_QEMU_MIPS) \ | ||||||
|   || defined (GRUB_MACHINE_MIPS_LOONGSON) || defined (GRUB_MACHINE_ARC) \ |   || defined (GRUB_MACHINE_MIPS_LOONGSON) || defined (GRUB_MACHINE_ARC) \ | ||||||
|   || defined (__sparc__) |   || defined (__sparc__) || defined (GRUB_MACHINE_UBOOT) | ||||||
| /* FIXME: stack is between 2 heap regions. Move it.  */ | /* FIXME: stack is between 2 heap regions. Move it.  */ | ||||||
| #define GRUB_KERNEL_PRELOAD_SPACE_REUSABLE 1 | #define GRUB_KERNEL_PRELOAD_SPACE_REUSABLE 1 | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | @ -112,3 +112,14 @@ void EXPORT_FUNC (_savegpr_29) (void); | ||||||
| void EXPORT_FUNC (_savegpr_30) (void); | void EXPORT_FUNC (_savegpr_30) (void); | ||||||
| void EXPORT_FUNC (_savegpr_31) (void); | void EXPORT_FUNC (_savegpr_31) (void); | ||||||
| #endif | #endif | ||||||
|  | 
 | ||||||
|  | #if defined (__arm__) | ||||||
|  | void EXPORT_FUNC (__aeabi_idiv) (void); | ||||||
|  | void EXPORT_FUNC (__aeabi_idivmod) (void); | ||||||
|  | void EXPORT_FUNC (__aeabi_lasr) (void); | ||||||
|  | void EXPORT_FUNC (__aeabi_llsl) (void); | ||||||
|  | void EXPORT_FUNC (__aeabi_llsr) (void); | ||||||
|  | void EXPORT_FUNC (__aeabi_uidiv) (void); | ||||||
|  | void EXPORT_FUNC (__aeabi_uidivmod) (void); | ||||||
|  | void EXPORT_FUNC (__wrap___clear_cache) (void *, void *); | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | @ -103,6 +103,7 @@ | ||||||
| #define GRUB_KERNEL_I386_IEEE1275_MOD_GAP 0x0 | #define GRUB_KERNEL_I386_IEEE1275_MOD_GAP 0x0 | ||||||
| #define GRUB_KERNEL_I386_COREBOOT_MOD_GAP 0x0 | #define GRUB_KERNEL_I386_COREBOOT_MOD_GAP 0x0 | ||||||
| #define GRUB_KERNEL_SPARC64_IEEE1275_MOD_GAP 0x0 | #define GRUB_KERNEL_SPARC64_IEEE1275_MOD_GAP 0x0 | ||||||
|  | #define GRUB_KERNEL_ARM_UBOOT_MOD_GAP 0x0 | ||||||
| 
 | 
 | ||||||
| #define GRUB_KERNEL_POWERPC_IEEE1275_MOD_ALIGN 0x1000 | #define GRUB_KERNEL_POWERPC_IEEE1275_MOD_ALIGN 0x1000 | ||||||
| #define GRUB_KERNEL_SPARC64_IEEE1275_MOD_ALIGN 0x1 | #define GRUB_KERNEL_SPARC64_IEEE1275_MOD_ALIGN 0x1 | ||||||
|  | @ -111,6 +112,10 @@ | ||||||
| #define GRUB_KERNEL_MIPS_ARC_MOD_ALIGN 0x1 | #define GRUB_KERNEL_MIPS_ARC_MOD_ALIGN 0x1 | ||||||
| #define GRUB_KERNEL_MIPS_QEMU_MIPS_MOD_ALIGN 0x1 | #define GRUB_KERNEL_MIPS_QEMU_MIPS_MOD_ALIGN 0x1 | ||||||
| 
 | 
 | ||||||
|  | #define GRUB_KERNEL_ARM_UBOOT_MOD_ALIGN 	0x8 | ||||||
|  | #define GRUB_KERNEL_ARM_UBOOT_TOTAL_MODULE_SIZE	0x4 | ||||||
|  | #define GRUB_KERNEL_ARM_UBOOT_LINK_ADDR		0x08000000 | ||||||
|  | 
 | ||||||
| /* Minimal gap between _end and the start of the modules.  It's a hack
 | /* Minimal gap between _end and the start of the modules.  It's a hack
 | ||||||
|    for PowerMac to prevent "CLAIM failed" error.  The real fix is to |    for PowerMac to prevent "CLAIM failed" error.  The real fix is to | ||||||
|    rewrite grub-mkimage to generate valid ELF files.  */ |    rewrite grub-mkimage to generate valid ELF files.  */ | ||||||
|  |  | ||||||
|  | @ -29,7 +29,11 @@ | ||||||
| 
 | 
 | ||||||
| #if HAVE_ASM_USCORE | #if HAVE_ASM_USCORE | ||||||
| #ifdef ASM_FILE | #ifdef ASM_FILE | ||||||
| # define EXT_C(sym)	_ ## sym | # ifndef (__arm__) | ||||||
|  | #  define EXT_C(sym)	_ ## sym | ||||||
|  | # else | ||||||
|  | #  define EXT_C(sym)	% ## sym | ||||||
|  | # endif | ||||||
| #else | #else | ||||||
| # define EXT_C(sym)	"_" sym | # define EXT_C(sym)	"_" sym | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
							
								
								
									
										184
									
								
								include/grub/uboot/api_public.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										184
									
								
								include/grub/uboot/api_public.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,184 @@ | ||||||
|  | /*
 | ||||||
|  |  * (C) Copyright 2007-2008 Semihalf | ||||||
|  |  * | ||||||
|  |  * Written by: Rafal Jaworowski <raj@semihalf.com> | ||||||
|  |  * | ||||||
|  |  * This file is dual licensed; you can use it under the terms of | ||||||
|  |  * either the GPL, or the BSD license, at your option. | ||||||
|  |  * | ||||||
|  |  * I. GPL: | ||||||
|  |  * | ||||||
|  |  * This file 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 2 of | ||||||
|  |  * the License, or (at your option) any later version. | ||||||
|  |  * | ||||||
|  |  * This file 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 this program; if not, write to the Free Software | ||||||
|  |  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | ||||||
|  |  * MA 02111-1307 USA | ||||||
|  |  * | ||||||
|  |  * Alternatively, | ||||||
|  |  * | ||||||
|  |  * II. BSD license: | ||||||
|  |  * | ||||||
|  |  * 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. | ||||||
|  |  * | ||||||
|  |  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR 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 AUTHOR 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. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef _API_PUBLIC_H_ | ||||||
|  | #define _API_PUBLIC_H_ | ||||||
|  | 
 | ||||||
|  | #define API_EINVAL		1	/* invalid argument(s)  */ | ||||||
|  | #define API_ENODEV		2	/* no device            */ | ||||||
|  | #define API_ENOMEM		3	/* no memory            */ | ||||||
|  | #define API_EBUSY		4	/* busy, occupied etc.  */ | ||||||
|  | #define API_EIO			5	/* I/O error            */ | ||||||
|  | #define API_ESYSC		6	/* syscall error        */ | ||||||
|  | 
 | ||||||
|  | typedef int (*scp_t) (int, int *, ...); | ||||||
|  | 
 | ||||||
|  | typedef grub_uint16_t uint16_t; | ||||||
|  | typedef grub_uint32_t uint32_t; | ||||||
|  | 
 | ||||||
|  | #define API_SIG_VERSION	1 | ||||||
|  | #define API_SIG_MAGIC	"UBootAPI" | ||||||
|  | #define API_SIG_MAGLEN	8 | ||||||
|  | 
 | ||||||
|  | struct api_signature | ||||||
|  | { | ||||||
|  |   char magic[API_SIG_MAGLEN];	/* magic string */ | ||||||
|  |   uint16_t version;		/* API version */ | ||||||
|  |   uint32_t checksum;		/* checksum of this sig struct */ | ||||||
|  |   scp_t syscall;		/* entry point to the API */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | enum | ||||||
|  | { | ||||||
|  |   API_RSVD = 0, | ||||||
|  |   API_GETC, | ||||||
|  |   API_PUTC, | ||||||
|  |   API_TSTC, | ||||||
|  |   API_PUTS, | ||||||
|  |   API_RESET, | ||||||
|  |   API_GET_SYS_INFO, | ||||||
|  |   API_UDELAY, | ||||||
|  |   API_GET_TIMER, | ||||||
|  |   API_DEV_ENUM, | ||||||
|  |   API_DEV_OPEN, | ||||||
|  |   API_DEV_CLOSE, | ||||||
|  |   API_DEV_READ, | ||||||
|  |   API_DEV_WRITE, | ||||||
|  |   API_ENV_ENUM, | ||||||
|  |   API_ENV_GET, | ||||||
|  |   API_ENV_SET, | ||||||
|  |   API_DISPLAY_GET_INFO, | ||||||
|  |   API_DISPLAY_DRAW_BITMAP, | ||||||
|  |   API_DISPLAY_CLEAR, | ||||||
|  |   API_MAXCALL | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define MR_ATTR_FLASH	0x0001 | ||||||
|  | #define MR_ATTR_DRAM	0x0002 | ||||||
|  | #define MR_ATTR_SRAM	0x0003 | ||||||
|  | #define MR_ATTR_MASK	0x000f | ||||||
|  | 
 | ||||||
|  | struct mem_region | ||||||
|  | { | ||||||
|  |   unsigned long start; | ||||||
|  |   unsigned long size; | ||||||
|  |   int flags; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct sys_info | ||||||
|  | { | ||||||
|  |   unsigned long clk_bus; | ||||||
|  |   unsigned long clk_cpu; | ||||||
|  |   unsigned long bar; | ||||||
|  |   struct mem_region *mr; | ||||||
|  |   int mr_no;			/* number of memory regions */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #undef CONFIG_SYS_64BIT_LBA | ||||||
|  | #ifdef CONFIG_SYS_64BIT_LBA | ||||||
|  | typedef u_int64_t lbasize_t; | ||||||
|  | #else | ||||||
|  | typedef unsigned long lbasize_t; | ||||||
|  | #endif | ||||||
|  | typedef unsigned long lbastart_t; | ||||||
|  | 
 | ||||||
|  | #define DEV_TYP_NONE	0x0000 | ||||||
|  | #define DEV_TYP_NET	0x0001 | ||||||
|  | 
 | ||||||
|  | #define DEV_TYP_STOR	0x0002 | ||||||
|  | #define DT_STOR_IDE	0x0010 | ||||||
|  | #define DT_STOR_SCSI	0x0020 | ||||||
|  | #define DT_STOR_USB	0x0040 | ||||||
|  | #define DT_STOR_MMC	0x0080 | ||||||
|  | #define DT_STOR_SATA	0x0100 | ||||||
|  | 
 | ||||||
|  | #define DEV_STA_CLOSED	0x0000	/* invalid, closed */ | ||||||
|  | #define DEV_STA_OPEN	0x0001	/* open i.e. active */ | ||||||
|  | 
 | ||||||
|  | struct device_info | ||||||
|  | { | ||||||
|  |   int type; | ||||||
|  |   void *cookie; | ||||||
|  | 
 | ||||||
|  |   union | ||||||
|  |   { | ||||||
|  |     struct | ||||||
|  |     { | ||||||
|  |       lbasize_t block_count;	/* no of blocks */ | ||||||
|  |       unsigned long block_size;	/* size of one block */ | ||||||
|  |     } storage; | ||||||
|  | 
 | ||||||
|  |     struct | ||||||
|  |     { | ||||||
|  |       unsigned char hwaddr[6]; | ||||||
|  |     } net; | ||||||
|  |   } info; | ||||||
|  | #define di_stor info.storage | ||||||
|  | #define di_net info.net | ||||||
|  | 
 | ||||||
|  |   int state; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define DISPLAY_TYPE_LCD	0x0001 | ||||||
|  | #define DISPLAY_TYPE_VIDEO	0x0002 | ||||||
|  | 
 | ||||||
|  | struct display_info | ||||||
|  | { | ||||||
|  |   int type; | ||||||
|  |   /* screen size in pixels */ | ||||||
|  |   int pixel_width; | ||||||
|  |   int pixel_height; | ||||||
|  |   /* screen size in rows and columns of text */ | ||||||
|  |   int screen_rows; | ||||||
|  |   int screen_cols; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #endif /* _API_PUBLIC_H_ */ | ||||||
							
								
								
									
										29
									
								
								include/grub/uboot/console.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								include/grub/uboot/console.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,29 @@ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/>.
 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef GRUB_CONSOLE_MACHINE_HEADER | ||||||
|  | #define GRUB_CONSOLE_MACHINE_HEADER 1 | ||||||
|  | 
 | ||||||
|  | /* Initialize the console system.  */ | ||||||
|  | void grub_console_init_early (void); | ||||||
|  | void grub_console_init_lately (void); | ||||||
|  | 
 | ||||||
|  | /* Exit the console system.  */ | ||||||
|  | void grub_console_fini (void); | ||||||
|  | 
 | ||||||
|  | #endif /* ! GRUB_CONSOLE_MACHINE_HEADER */ | ||||||
							
								
								
									
										44
									
								
								include/grub/uboot/disk.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								include/grub/uboot/disk.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,44 @@ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013  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/>.
 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef GRUB_UBOOT_DISK_HEADER | ||||||
|  | #define GRUB_UBOOT_DISK_HEADER	1 | ||||||
|  | 
 | ||||||
|  | #include <grub/symbol.h> | ||||||
|  | #include <grub/disk.h> | ||||||
|  | #include <grub/uboot/uboot.h> | ||||||
|  | 
 | ||||||
|  | void grub_ubootdisk_init (void); | ||||||
|  | void grub_ubootdisk_fini (void); | ||||||
|  | 
 | ||||||
|  | enum disktype | ||||||
|  | { cd, fd, hd }; | ||||||
|  | 
 | ||||||
|  | struct ubootdisk_data | ||||||
|  | { | ||||||
|  |   struct ubootdisk_data *next; | ||||||
|  |   void *cookie; | ||||||
|  |   int handle; | ||||||
|  |   int opencount; | ||||||
|  |   enum disktype type; | ||||||
|  |   grub_uint32_t block_size; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | grub_err_t grub_ubootdisk_register (struct device_info *newdev, int handle); | ||||||
|  | 
 | ||||||
|  | #endif /* ! GRUB_UBOOT_DISK_HEADER */ | ||||||
							
								
								
									
										150
									
								
								include/grub/uboot/uboot.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										150
									
								
								include/grub/uboot/uboot.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,150 @@ | ||||||
|  | /* uboot.h - declare variables and functions for U-Boot support */ | ||||||
|  | /*
 | ||||||
|  |  *  GRUB  --  GRand Unified Bootloader | ||||||
|  |  *  Copyright (C) 2013 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/>.
 | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef GRUB_UBOOT_UBOOT_HEADER | ||||||
|  | #define GRUB_UBOOT_UBOOT_HEADER	1 | ||||||
|  | 
 | ||||||
|  | #include <grub/types.h> | ||||||
|  | #include <grub/dl.h> | ||||||
|  | 
 | ||||||
|  | /* Functions.  */ | ||||||
|  | void grub_uboot_mm_init (void); | ||||||
|  | void grub_uboot_init (void); | ||||||
|  | void grub_uboot_fini (void); | ||||||
|  | 
 | ||||||
|  | void uboot_return (int) __attribute__ ((noreturn)); | ||||||
|  | 
 | ||||||
|  | grub_addr_t uboot_get_real_bss_start (void); | ||||||
|  | 
 | ||||||
|  | grub_err_t grub_uboot_probe_hardware (void); | ||||||
|  | 
 | ||||||
|  | extern grub_addr_t EXPORT_VAR (start_of_ram); | ||||||
|  | 
 | ||||||
|  | grub_uint32_t EXPORT_FUNC (uboot_get_machine_type) (void); | ||||||
|  | grub_addr_t EXPORT_FUNC (uboot_get_boot_data) (void); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * The U-Boot API operates through a "syscall" interface, consisting of an | ||||||
|  |  * entry point address and a set of syscall numbers. The location of this | ||||||
|  |  * entry point is described in a structure allocated on the U-Boot heap. | ||||||
|  |  * We scan through a defined region around the hint address passed to us | ||||||
|  |  * from U-Boot. | ||||||
|  |  */ | ||||||
|  | #include <grub/uboot/api_public.h> | ||||||
|  | 
 | ||||||
|  | #define UBOOT_API_SEARCH_LEN (3 * 1024 * 1024) | ||||||
|  | int uboot_api_init (void); | ||||||
|  | 
 | ||||||
|  | /* All functions below are wrappers around the uboot_syscall() function */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_getc(int *c) | ||||||
|  |  */ | ||||||
|  | int uboot_getc (void); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_tstc(int *c) | ||||||
|  |  */ | ||||||
|  | int uboot_tstc (void); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_putc(char *ch) | ||||||
|  |  */ | ||||||
|  | void uboot_putc (int c); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_puts(const char *s) | ||||||
|  |  */ | ||||||
|  | void uboot_puts (const char *s); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_reset(void) | ||||||
|  |  */ | ||||||
|  | void EXPORT_FUNC (uboot_reset) (void); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_get_sys_info(struct sys_info *si) | ||||||
|  |  * | ||||||
|  |  * fill out the sys_info struct containing selected parameters about the | ||||||
|  |  * machine | ||||||
|  |  */ | ||||||
|  | struct sys_info *uboot_get_sys_info (void); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_udelay(unsigned long *udelay) | ||||||
|  |  */ | ||||||
|  | void uboot_udelay (grub_uint32_t usec); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_get_timer(unsigned long *current, unsigned long *base) | ||||||
|  |  */ | ||||||
|  | grub_uint32_t uboot_get_timer (grub_uint32_t base); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_dev_enum(struct device_info *) | ||||||
|  |  */ | ||||||
|  | int uboot_dev_enum (void); | ||||||
|  | 
 | ||||||
|  | struct device_info *uboot_dev_get (int handle); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_dev_open(struct device_info *) | ||||||
|  |  */ | ||||||
|  | int uboot_dev_open (int handle); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_dev_close(struct device_info *) | ||||||
|  |  */ | ||||||
|  | int uboot_dev_close (int handle); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Notice: this is for sending network packets only, as U-Boot does not | ||||||
|  |  * support writing to storage at the moment (12.2007) | ||||||
|  |  * | ||||||
|  |  * int API_dev_write(struct device_info *di, void *buf,	int *len) | ||||||
|  |  */ | ||||||
|  | int uboot_dev_write (int handle, void *buf, int *len); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_dev_read( | ||||||
|  |  *	struct device_info *di, | ||||||
|  |  *	void *buf, | ||||||
|  |  *	size_t *len, | ||||||
|  |  *	unsigned long *start | ||||||
|  |  *	size_t *act_len | ||||||
|  |  * ) | ||||||
|  |  */ | ||||||
|  | int uboot_dev_read (int handle, void *buf, lbasize_t blocks, | ||||||
|  | 		    lbastart_t start, lbasize_t * real_blocks); | ||||||
|  | 
 | ||||||
|  | int uboot_dev_recv (int handle, void *buf, int size, int *real_size); | ||||||
|  | int uboot_dev_send (int handle, void *buf, int size); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_env_get(const char *name, char **value) | ||||||
|  |  */ | ||||||
|  | char *uboot_env_get (const char *name); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * int API_env_set(const char *name, const char *value) | ||||||
|  |  */ | ||||||
|  | void uboot_env_set (const char *name, const char *value); | ||||||
|  | 
 | ||||||
|  | #endif /* ! GRUB_UBOOT_UBOOT_HEADER */ | ||||||
|  | @ -316,6 +316,8 @@ if [ x$source_dir = x ]; then | ||||||
| 		    target=i386-pc | 		    target=i386-pc | ||||||
| 		fi | 		fi | ||||||
| 		;; | 		;; | ||||||
|  | 	    x"arm"*) | ||||||
|  | 		target="arm-uboot";; | ||||||
| 	    *) | 	    *) | ||||||
| 		gettext "Unable to determine your platform. Use --target." ; | 		gettext "Unable to determine your platform. Use --target." ; | ||||||
| 		echo	;; | 		echo	;; | ||||||
|  | @ -335,7 +337,7 @@ if [ "${grub_modinfo_target_cpu}-${grub_modinfo_platform}" = "i386-pc" ] ; then | ||||||
|     if [ x$disk_module = xunspecified ]; then |     if [ x$disk_module = xunspecified ]; then | ||||||
| 	disk_module=biosdisk | 	disk_module=biosdisk | ||||||
|     fi |     fi | ||||||
| elif [ "${grub_modinfo_platform}" = "ieee1275" ] || [ "${grub_modinfo_platform}" = "efi" ] || [ "${grub_modinfo_platform}" = "arc" ] ; then | elif [ "${grub_modinfo_platform}" = "ieee1275" ] || [ "${grub_modinfo_platform}" = "efi" ] || [ "${grub_modinfo_platform}" = "arc" ] || [ "${grub_modinfo_platform}" = "uboot" ] ; then | ||||||
|     disk_module= |     disk_module= | ||||||
| else | else | ||||||
|     disk_module=native |     disk_module=native | ||||||
|  | @ -464,6 +466,8 @@ if [ x"$grub_modinfo_platform" = xefi ]; then | ||||||
| 	    # expansion. | 	    # expansion. | ||||||
| 		ia64) | 		ia64) | ||||||
| 		    efi_file=BOOTIA64.EFI ;; | 		    efi_file=BOOTIA64.EFI ;; | ||||||
|  | 		arm) | ||||||
|  | 		    efi_file=BOOTARM.EFI ;; | ||||||
| 	    esac | 	    esac | ||||||
| 	else | 	else | ||||||
| 	    # It is convenient for each architecture to have a different | 	    # It is convenient for each architecture to have a different | ||||||
|  | @ -478,6 +482,8 @@ if [ x"$grub_modinfo_platform" = xefi ]; then | ||||||
| 	 # expansion. | 	 # expansion. | ||||||
| 		ia64) | 		ia64) | ||||||
| 		    efi_file=grubia64.efi ;; | 		    efi_file=grubia64.efi ;; | ||||||
|  | 		arm) | ||||||
|  | 		    efi_file=grubarm.efi ;; | ||||||
| 		*) | 		*) | ||||||
| 		    efi_file=grub.efi ;; | 		    efi_file=grub.efi ;; | ||||||
| 	    esac | 	    esac | ||||||
|  | @ -827,6 +833,14 @@ elif [ x"$grub_modinfo_platform" = xefi ]; then | ||||||
| 		-L "$bootloader_id" -l "\\EFI\\$efi_distributor\\$efi_file" | 		-L "$bootloader_id" -l "\\EFI\\$efi_distributor\\$efi_file" | ||||||
| 	fi | 	fi | ||||||
|     fi |     fi | ||||||
|  | elif [ x"${grub_modinfo_target_cpu}-${grub_modinfo_platform}" = xarm-uboot ]; then | ||||||
|  |     grub_imgname="${grubdir}/${grub_modinfo_target_cpu}-$grub_modinfo_platform/core.${imgext}" | ||||||
|  |     raw_imgname="${uboot_imgname}.raw" | ||||||
|  |     mv "$grub_imgname" "$raw_imgname" | ||||||
|  |     mkimage -T kernel -A ARM -O Linux -a 0x08000000 -e 0x08000000 -C none -d "$raw_imgname" "$grub_imgname" | ||||||
|  |     if [ $? -eq 0 ]; then | ||||||
|  | 	rm -f "$raw_imgname" | ||||||
|  |     fi | ||||||
| else | else | ||||||
|     gettext "WARNING: no platform-specific install was performed" 1>&2 |     gettext "WARNING: no platform-specific install was performed" 1>&2 | ||||||
|     echo 1>&2 |     echo 1>&2 | ||||||
|  |  | ||||||
|  | @ -69,7 +69,7 @@ struct image_target_desc | ||||||
|     IMAGE_SPARC64_AOUT, IMAGE_SPARC64_RAW, IMAGE_I386_IEEE1275, |     IMAGE_SPARC64_AOUT, IMAGE_SPARC64_RAW, IMAGE_I386_IEEE1275, | ||||||
|     IMAGE_LOONGSON_ELF, IMAGE_QEMU, IMAGE_PPC, IMAGE_YEELOONG_FLASH, |     IMAGE_LOONGSON_ELF, IMAGE_QEMU, IMAGE_PPC, IMAGE_YEELOONG_FLASH, | ||||||
|     IMAGE_FULOONG2F_FLASH, IMAGE_I386_PC_PXE, IMAGE_MIPS_ARC, |     IMAGE_FULOONG2F_FLASH, IMAGE_I386_PC_PXE, IMAGE_MIPS_ARC, | ||||||
|     IMAGE_QEMU_MIPS_FLASH |     IMAGE_QEMU_MIPS_FLASH, IMAGE_UBOOT | ||||||
|   } id; |   } id; | ||||||
|   enum |   enum | ||||||
|     { |     { | ||||||
|  | @ -455,6 +455,46 @@ struct image_target_desc image_targets[] = | ||||||
|       .link_align = GRUB_KERNEL_MIPS_QEMU_MIPS_LINK_ALIGN, |       .link_align = GRUB_KERNEL_MIPS_QEMU_MIPS_LINK_ALIGN, | ||||||
|       .default_compression = COMPRESSION_NONE |       .default_compression = COMPRESSION_NONE | ||||||
|     }, |     }, | ||||||
|  |     { | ||||||
|  |       .dirname = "arm-uboot", | ||||||
|  |       .names = { "arm-uboot", NULL }, | ||||||
|  |       .voidp_sizeof = 4, | ||||||
|  |       .bigendian = 0, | ||||||
|  |       .id = IMAGE_UBOOT,  | ||||||
|  |       .flags = PLATFORM_FLAGS_NONE, | ||||||
|  |       .total_module_size = GRUB_KERNEL_ARM_UBOOT_TOTAL_MODULE_SIZE, | ||||||
|  |       .decompressor_compressed_size = TARGET_NO_FIELD, | ||||||
|  |       .decompressor_uncompressed_size = TARGET_NO_FIELD, | ||||||
|  |       .decompressor_uncompressed_addr = TARGET_NO_FIELD, | ||||||
|  |       .section_align = GRUB_KERNEL_ARM_UBOOT_MOD_ALIGN, | ||||||
|  |       .vaddr_offset = 0, | ||||||
|  |       .link_addr = GRUB_KERNEL_ARM_UBOOT_LINK_ADDR, | ||||||
|  |       .elf_target = EM_ARM, | ||||||
|  |       .mod_gap = GRUB_KERNEL_ARM_UBOOT_MOD_GAP, | ||||||
|  |       .mod_align = GRUB_KERNEL_ARM_UBOOT_MOD_ALIGN, | ||||||
|  |       .link_align = 4 | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       .dirname = "arm-efi", | ||||||
|  |       .names = { "arm-efi", NULL }, | ||||||
|  |       .voidp_sizeof = 4, | ||||||
|  |       .bigendian = 0,  | ||||||
|  |       .id = IMAGE_EFI,  | ||||||
|  |       .flags = PLATFORM_FLAGS_NONE, | ||||||
|  |       .total_module_size = TARGET_NO_FIELD, | ||||||
|  |       .decompressor_compressed_size = TARGET_NO_FIELD, | ||||||
|  |       .decompressor_uncompressed_size = TARGET_NO_FIELD, | ||||||
|  |       .decompressor_uncompressed_addr = TARGET_NO_FIELD, | ||||||
|  |       .section_align = GRUB_PE32_SECTION_ALIGNMENT, | ||||||
|  |       .vaddr_offset = ALIGN_UP (GRUB_PE32_MSDOS_STUB_SIZE | ||||||
|  |                                 + GRUB_PE32_SIGNATURE_SIZE | ||||||
|  |                                 + sizeof (struct grub_pe32_coff_header) | ||||||
|  |                                 + sizeof (struct grub_pe32_optional_header) | ||||||
|  |                                 + 4 * sizeof (struct grub_pe32_section_table), | ||||||
|  |                                 GRUB_PE32_SECTION_ALIGNMENT), | ||||||
|  |       .pe_target = GRUB_PE32_MACHINE_ARMTHUMB_MIXED, | ||||||
|  |       .elf_target = EM_ARM, | ||||||
|  |     }, | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
| #define grub_target_to_host32(x) (grub_target_to_host32_real (image_target, (x))) | #define grub_target_to_host32(x) (grub_target_to_host32_real (image_target, (x))) | ||||||
|  | @ -1022,6 +1062,7 @@ generate_image (const char *dir, const char *prefix, | ||||||
|     case IMAGE_SPARC64_RAW: |     case IMAGE_SPARC64_RAW: | ||||||
|     case IMAGE_I386_IEEE1275: |     case IMAGE_I386_IEEE1275: | ||||||
|     case IMAGE_PPC: |     case IMAGE_PPC: | ||||||
|  |     case IMAGE_UBOOT: | ||||||
|       break; |       break; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -1684,6 +1725,9 @@ generate_image (const char *dir, const char *prefix, | ||||||
| 	core_size = program_size + header_size + footer_size; | 	core_size = program_size + header_size + footer_size; | ||||||
|       } |       } | ||||||
|       break; |       break; | ||||||
|  |     case IMAGE_UBOOT: | ||||||
|  |       /* Raw image, header added by grub-install */ | ||||||
|  |       break; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|   grub_util_write_image (core_img, core_size, out, outname); |   grub_util_write_image (core_img, core_size, out, outname); | ||||||
|  |  | ||||||
|  | @ -58,6 +58,11 @@ | ||||||
| #error "I'm confused" | #error "I'm confused" | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
|  | static Elf_Addr SUFFIX (entry_point); | ||||||
|  | 
 | ||||||
|  | grub_err_t reloc_thm_call (grub_uint16_t *addr, Elf32_Addr sym_addr); | ||||||
|  | grub_err_t reloc_thm_jump19 (grub_uint16_t *addr, Elf32_Addr sym_addr); | ||||||
|  | 
 | ||||||
| /* Relocate symbols; note that this function overwrites the symbol table.
 | /* Relocate symbols; note that this function overwrites the symbol table.
 | ||||||
|    Return the address of a start symbol.  */ |    Return the address of a start symbol.  */ | ||||||
| static Elf_Addr | static Elf_Addr | ||||||
|  | @ -528,6 +533,48 @@ SUFFIX (relocate_addresses) (Elf_Ehdr *e, Elf_Shdr *sections, | ||||||
| 		} | 		} | ||||||
| 	       break; | 	       break; | ||||||
| #endif | #endif | ||||||
|  | #if defined(MKIMAGE_ELF32) | ||||||
|  | 	     case EM_ARM: | ||||||
|  | 	       { | ||||||
|  | 		 sym_addr += addend; | ||||||
|  | 		 sym_addr -= SUFFIX (entry_point); | ||||||
|  | 		 switch (ELF_R_TYPE (info)) | ||||||
|  | 		   { | ||||||
|  | 		   case R_ARM_ABS32: | ||||||
|  | 		     { | ||||||
|  | 		       grub_util_info ("  ABS32:\toffset=%d\t(0x%08x)", | ||||||
|  | 				       (int) sym_addr, (int) sym_addr); | ||||||
|  | 		       /* Data will be naturally aligned */ | ||||||
|  | 		       //      sym_addr -= offset;
 | ||||||
|  | 		       sym_addr += 0x400; | ||||||
|  | 		       *target = grub_host_to_target32 (grub_target_to_host32 (*target) + sym_addr); | ||||||
|  | 		     } | ||||||
|  | 		     break; | ||||||
|  | 		   case R_ARM_THM_CALL: | ||||||
|  | 		   case R_ARM_THM_JUMP24: | ||||||
|  | 		     { | ||||||
|  | 		       grub_util_info ("  THM_JUMP24:\ttarget=0x%08x\toffset=(0x%08x)",	(unsigned int) target, sym_addr); | ||||||
|  | 		       sym_addr -= offset; | ||||||
|  | 		       /* Thumb instructions can be 16-bit aligned */ | ||||||
|  | 		       reloc_thm_call ((grub_uint16_t *) target, sym_addr); | ||||||
|  | 		     } | ||||||
|  | 		     break; | ||||||
|  | 		   case R_ARM_THM_JUMP19: | ||||||
|  | 		     { | ||||||
|  | 		       grub_util_info ("  THM_JUMP19:\toffset=%d\t(0x%08x)", | ||||||
|  | 				       sym_addr, sym_addr); | ||||||
|  | 		       sym_addr -= offset; | ||||||
|  | 		       /* Thumb instructions can be 16-bit aligned */ | ||||||
|  | 		       reloc_thm_jump19 ((grub_uint16_t *) target, sym_addr); | ||||||
|  | 		     } | ||||||
|  | 		     break; | ||||||
|  | 		   default: | ||||||
|  | 		     grub_util_error (_("relocation 0x%x is not implemented yet!"), ELF_R_TYPE (info)); | ||||||
|  | 		     break; | ||||||
|  | 		   } | ||||||
|  | 		 break; | ||||||
|  | 	       } | ||||||
|  | #endif /* MKIMAGE_ELF32 */ | ||||||
| 	     default: | 	     default: | ||||||
| 	       grub_util_error ("unknown architecture type %d", | 	       grub_util_error ("unknown architecture type %d", | ||||||
| 				image_target->elf_target); | 				image_target->elf_target); | ||||||
|  | @ -755,6 +802,46 @@ SUFFIX (make_reloc_section) (Elf_Ehdr *e, void **out, | ||||||
| 		  break; | 		  break; | ||||||
| 		} | 		} | ||||||
| 		break; | 		break; | ||||||
|  | #if defined(MKIMAGE_ELF32) | ||||||
|  | 	      case EM_ARM: | ||||||
|  | 		switch (ELF_R_TYPE (info)) | ||||||
|  | 		  { | ||||||
|  | 		    /* Relative relocations do not require fixup entries. */ | ||||||
|  | 		  case R_ARM_JUMP24: | ||||||
|  | 		  case R_ARM_THM_CALL: | ||||||
|  | 		  case R_ARM_THM_JUMP19: | ||||||
|  | 		  case R_ARM_THM_JUMP24: | ||||||
|  | 		    { | ||||||
|  | 		      Elf_Addr addr; | ||||||
|  | 
 | ||||||
|  | 		      addr = section_address + offset; | ||||||
|  | 		      grub_util_info ("  %s:  not adding fixup: 0x%08x : 0x%08x", __FUNCTION__, (unsigned int) addr, (unsigned int) current_address); | ||||||
|  | 		    } | ||||||
|  | 		    break; | ||||||
|  | 		    /* Create fixup entry for PE/COFF loader */ | ||||||
|  | 		  case R_ARM_ABS32: | ||||||
|  | 		    {       | ||||||
|  | 		      Elf_Addr addr; | ||||||
|  | 
 | ||||||
|  | 		      addr = section_address + offset; | ||||||
|  | #if 0 | ||||||
|  | 		      grub_util_info ("  %s:  add_fixup: 0x%08x : 0x%08x", | ||||||
|  | 				      __FUNCTION__, (unsigned int) addr, | ||||||
|  | 				      (unsigned int) current_address); | ||||||
|  | #endif | ||||||
|  | 		      current_address | ||||||
|  | 			= SUFFIX (add_fixup_entry) (&lst, | ||||||
|  | 						    GRUB_PE32_REL_BASED_HIGHLOW, | ||||||
|  | 						    addr, 0, current_address, | ||||||
|  | 						    image_target); | ||||||
|  | 		    } | ||||||
|  | 		    break; | ||||||
|  | 		  default: | ||||||
|  | 		    grub_util_error (_("relocation 0x%x is not implemented yet2"), ELF_R_TYPE (info)); | ||||||
|  | 		    break; | ||||||
|  | 		  } | ||||||
|  | 		break; | ||||||
|  | #endif /* defined(MKIMAGE_ELF32) */ | ||||||
| 	      default: | 	      default: | ||||||
| 		grub_util_error ("unknown machine type 0x%x", image_target->elf_target); | 		grub_util_error ("unknown machine type 0x%x", image_target->elf_target); | ||||||
| 	      } | 	      } | ||||||
|  | @ -1065,6 +1152,8 @@ SUFFIX (load_image) (const char *kernel_path, grub_size_t *exec_size, | ||||||
|       if (*start == 0) |       if (*start == 0) | ||||||
| 	grub_util_error ("start symbol is not defined"); | 	grub_util_error ("start symbol is not defined"); | ||||||
|        |        | ||||||
|  |       SUFFIX (entry_point) = (Elf_Addr) *start; | ||||||
|  | 
 | ||||||
|       /* Resolve addresses in the virtual address space.  */ |       /* Resolve addresses in the virtual address space.  */ | ||||||
|       SUFFIX (relocate_addresses) (e, sections, section_addresses,  |       SUFFIX (relocate_addresses) (e, sections, section_addresses,  | ||||||
| 				   section_entsize, | 				   section_entsize, | ||||||
|  |  | ||||||
							
								
								
									
										103
									
								
								util/import_libfdt.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										103
									
								
								util/import_libfdt.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,103 @@ | ||||||
|  | #* | ||||||
|  | #*  GRUB  --  GRand Unified Bootloader | ||||||
|  | #*  Copyright (C) 2013  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/>. | ||||||
|  | #* | ||||||
|  | 
 | ||||||
|  | import re | ||||||
|  | import sys | ||||||
|  | import os | ||||||
|  | import codecs | ||||||
|  | import datetime | ||||||
|  | 
 | ||||||
|  | if len (sys.argv) < 3: | ||||||
|  |     print ("Usage: %s SOURCE DESTINATION" % sys.argv[0]) | ||||||
|  |     exit (0) | ||||||
|  | dtcdir = sys.argv[1] | ||||||
|  | indir = os.path.join (dtcdir, "libfdt/") | ||||||
|  | outdir = os.path.join (sys.argv[2], "lib/dtc-grub/libfdt/") | ||||||
|  | try: | ||||||
|  |     os.makedirs (outdir) | ||||||
|  | except: | ||||||
|  |     print ("WARNING: %s already exists" % outdir) | ||||||
|  | 
 | ||||||
|  | conf = codecs.open (os.path.join ("grub-core/", "Makefile.libfdt.def"), "w", "utf-8") | ||||||
|  | conf.write ("AutoGen definitions Makefile.tpl;\n\n") | ||||||
|  | conf.write ("module = {\n") | ||||||
|  | conf.write ("  name = fdt;\n") | ||||||
|  | conf.write ("  common = lib/dtc-grub/libfdt/fdt.c;\n") | ||||||
|  | conf.write ("  common = lib/dtc-grub/libfdt/fdt_ro.c;\n") | ||||||
|  | conf.write ("  common = lib/dtc-grub/libfdt/fdt_rw.c;\n") | ||||||
|  | conf.write ("  common = lib/dtc-grub/libfdt/fdt_strerror.c;\n") | ||||||
|  | conf.write ("  common = lib/dtc-grub/libfdt/fdt_sw.c;\n") | ||||||
|  | conf.write ("  common = lib/dtc-grub/libfdt/fdt_wip.c;\n") | ||||||
|  | conf.write ("  cppflags = '$(CPPFLAGS_POSIX) $(CPPFLAGS_LIBFDT)';\n") | ||||||
|  | conf.write ("\n") | ||||||
|  | conf.write ("  enable = fdt;\n") | ||||||
|  | conf.write ("};\n") | ||||||
|  | 
 | ||||||
|  | conf.close (); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | libfdt_files = sorted (os.listdir (indir)) | ||||||
|  | chlog = "" | ||||||
|  | 
 | ||||||
|  | for libfdt_file in libfdt_files: | ||||||
|  |     infile = os.path.join (indir, (libfdt_file)) | ||||||
|  |     outfile = os.path.join (outdir, (libfdt_file)) | ||||||
|  | 
 | ||||||
|  |     if not re.match (".*\.[ch]$", libfdt_file): | ||||||
|  |         chlog = "%s	* %s: Removed\n" % (chlog, libfdt_file) | ||||||
|  |         continue | ||||||
|  | 
 | ||||||
|  | #    print ("file: %s, infile: %s, outfile: %s" % (libfdt_file, infile, outfile)) | ||||||
|  | 
 | ||||||
|  |     f = codecs.open (infile, "r", "utf-8") | ||||||
|  |     fw = codecs.open (outfile, "w", "utf-8") | ||||||
|  | 
 | ||||||
|  |     lineno = 1 | ||||||
|  | 
 | ||||||
|  |     fw.write ("/* This file was automatically imported with \n") | ||||||
|  |     fw.write ("   import_libfdt.py. Please don't modify it */\n") | ||||||
|  |     fw.write ("#include <grub/dl.h>\n") | ||||||
|  | 
 | ||||||
|  |     # libfdt is dual-licensed: BSD or GPLv2+ | ||||||
|  |     if re.match (".*\.c$", libfdt_file): | ||||||
|  |         fw.write ("GRUB_MOD_LICENSE (\"GPLv2+\");\n") | ||||||
|  | 
 | ||||||
|  |     lines = f.readlines() | ||||||
|  | 
 | ||||||
|  |     for line in lines: | ||||||
|  |         fw.write (line) | ||||||
|  |          | ||||||
|  |     f.close () | ||||||
|  |     fw.close () | ||||||
|  | 
 | ||||||
|  | patchfile = os.path.join (dtcdir, "libfdt-grub.diff") | ||||||
|  | #print "Patchfile: %s\n" % patchfile | ||||||
|  | ret = os.system("patch -d %s -p1 < %s" % (outdir, patchfile)) | ||||||
|  | if ret: | ||||||
|  |     chlog = "%s	* Applied Grub build patch\n" % chlog | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | dt = datetime.date.today () | ||||||
|  | fw = codecs.open (os.path.join (outdir, "ImportLog"), "w", "utf-8") | ||||||
|  | fw.write ("%04d-%02d-%02d  Automatic import tool\n" % \ | ||||||
|  |           (dt.year,dt.month, dt.day)) | ||||||
|  | fw.write ("\n") | ||||||
|  | fw.write ("	Imported libfdt to GRUB\n") | ||||||
|  | fw.write ("\n") | ||||||
|  | fw.write (chlog) | ||||||
|  | fw.close () | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue