The main difference is in the build system used. Note that (x86_64) UEFI applications, drivers etc are supposed to be in the PE file format, using the
Microsoft x64 calling convention.
EDKII can use various different compilers (e.g. MSVC, ICC or a cross-compiled GCC it will build itself) to produce executables in this format. It also comes with an extensive collection of support libraries (e.g. std C library, math library etc) and example applications and drivers (including a PS/2 mouse driver which is particularly useful). Its main disadvantages, IMHO, are its sheer size and the fact you need to use its own customized makefile format to build anything useful.
gnu-efi approaches this a different way: it uses the native gcc on your system, which it expects to produce binaries in ELF format, and then uses a cross-compiled objcopy to convert these to PE with customized linker scripts and section names. Of course, this doesn't fix the problem that the compiled code still expects to use the sysV calling convention, rather than the Microsoft one. To get around this, all calls to UEFI functions need to go through uefi_call_wrapper() to put the arguments in the right places.
I have come up with a third way, using a gcc cross-compiler. Unfortunately there is no gcc target along the lines of x86_64-efi etc, so you need to choose something close. The most appropriate one I have found is i686-pc-mingw32-gcc for 32-bit and x86_64-w64-mingw32-gcc for 64-bit. These are available as cygwin packages and should also be available pre-built on most linux distributions. You should use the header files from gnu-efi and invoke as:
Code: Select all
x86_64-w64-mingw32-gcc -Ignu-efi/inc -Ignu-efi/inc/x86_64 -Ignu-efi/inc/protocol -ffreestanding -Wall -Wextra <options to disable mmx/sse etc> -c -o kernel.o kernel.c
x86_64-w64-mingw32-gcc -nostdlib -Wl,-dll -shared -Wl,--subsystem,10 -e efi_main -o kernel.efi kernel.o
where 'gnu-efi' is the path the gnu-efi distribution. The main drawbacks of this method are that the 32-bit version appends leading underscores to names (you need to tell the compiler not to do this), and that it defines the preprocessor macro 'WINDOWS' which causes some 3rd party packages to expect windows.h to be available, when it clearly isn't in a freestanding boot environment. For an example of this method, see the code and Makefiles
here.
Regards,
John.