Cosmos/source/Indy.IL2CPU.Assembler.X86.Native/Assembler.cs
mterwoord_cp aa77f8e7df
2008-01-09 14:41:28 +00:00

152 lines
No EOL
6.9 KiB
C#

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
namespace Indy.IL2CPU.Assembler.X86.Native {
public class Assembler: X86.Assembler {
public const string BreakMethodName = "_CODE_REQUESTED_BREAK_";
public Assembler(Func<string, string> aGetStreamForGroup, bool aInMetalMode)
: base(aGetStreamForGroup, aInMetalMode) {
}
public Assembler(Func<string, string> aGetStreamForGroup)
: base(aGetStreamForGroup) {
}
protected override void EmitCodeSectionHeader(string aGroup, StreamWriter aOutputWriter) {
base.EmitCodeSectionHeader(aGroup, aOutputWriter);
aOutputWriter.WriteLine("section .text");
if (aGroup == MainGroup) {
aOutputWriter.WriteLine("global Kernel_Start");
aOutputWriter.WriteLine("Kernel_Start: ");
aOutputWriter.WriteLine("");
aOutputWriter.WriteLine("; MultiBoot-compliant loader (e.g. GRUB or X.exe) provides info in registers: ");
aOutputWriter.WriteLine("; EBX=multiboot_info ");
aOutputWriter.WriteLine("; EAX=0x2BADB002 - check if it's really Multiboot loader ");
aOutputWriter.WriteLine("");
aOutputWriter.WriteLine(" ;- copy mb info - some stuff for you ");
//aOutputWriter.WriteLine(" mov mb_info, ebx");
aOutputWriter.WriteLine(" add ebx, 4");
aOutputWriter.WriteLine(" mov dword eax, [ebx]");
aOutputWriter.WriteLine(" mov dword [MultiBootInfo_Memory_Low], eax");
aOutputWriter.WriteLine(" add ebx, 4");
aOutputWriter.WriteLine(" mov dword eax, [ebx]");
aOutputWriter.WriteLine(" mov dword [MultiBootInfo_Memory_High], eax");
aOutputWriter.WriteLine("");
aOutputWriter.WriteLine(" mov esp,Kernel_Stack ");
aOutputWriter.WriteLine("");
aOutputWriter.WriteLine("; some more startups todo");
aOutputWriter.WriteLine(" cli");
//aOutputWriter.WriteLine(" push ebx");
aOutputWriter.WriteLine(" xchg bx, bx");
aOutputWriter.WriteLine(" call " + EntryPointName);
aOutputWriter.WriteLine(" .loop:");
aOutputWriter.WriteLine(" cli");
aOutputWriter.WriteLine(" hlt");
aOutputWriter.WriteLine(" jmp .loop");
aOutputWriter.WriteLine(" ");
aOutputWriter.WriteLine(" " + BreakMethodName + ":");
aOutputWriter.WriteLine(" ret");
aOutputWriter.WriteLine(" ");
}
}
protected override void EmitDataSectionHeader(string aGroup, StreamWriter aOutputWriter) {
base.EmitDataSectionHeader(aGroup, aOutputWriter);
if (aGroup == MainGroup) {
aOutputWriter.WriteLine("section .data");
aOutputWriter.WriteLine("_start: ");
aOutputWriter.WriteLine("; multiboot header ");
aOutputWriter.WriteLine("MBFLAGS equ 0x03 ; 4KB aligned modules etc., full memory info, ");
aOutputWriter.WriteLine(" ; use special header (see below) ");
aOutputWriter.WriteLine("dd 0x1BADB002 ; multiboot signature ");
aOutputWriter.WriteLine("dd MBFLAGS ; 4kb page aligment for modules, supply memory info ");
aOutputWriter.WriteLine("dd -0x1BADB002-MBFLAGS ; checksum=-(FLAGS+0x1BADB002) ");
aOutputWriter.WriteLine("; other data - that is the additional (optional) header which helps to load ");
aOutputWriter.WriteLine("; the kernel. ");
//aOutputWriter.WriteLine(" dd _start ; header_addr ");
//aOutputWriter.WriteLine(" dd _start ; load_addr ");
//aOutputWriter.WriteLine(" dd _end_data ; load_end_addr ");
//aOutputWriter.WriteLine(" dd _end ; bss_end_addr ");
//aOutputWriter.WriteLine(" dd Kernel_Start ; entry ");
aOutputWriter.WriteLine("; end of header ");
//aOutputWriter.WriteLine("mb_info multiboot_info");
aOutputWriter.WriteLine("MultiBootInfo_Memory_High dd 0");
aOutputWriter.WriteLine("MultiBootInfo_Memory_Low dd 0");
}
}
protected override void EmitIDataSectionHeader(string aGroup, StreamWriter aOutputWriter) {
}
protected override void EmitDataSectionFooter(string aGroup, StreamWriter aOutputWriter) {
base.EmitDataSectionFooter(aGroup, aOutputWriter);
if (aGroup == MainGroup) {
aOutputWriter.WriteLine("");
aOutputWriter.WriteLine(";--- bss --- place r*, d* ? directives here, so that you'll have a BSS. ");
aOutputWriter.WriteLine("");
aOutputWriter.WriteLine("Before_Kernel_Stack:");
//aOutputWriter.WriteLine("times 50000 resb 0 ; our own stack ");
aOutputWriter.WriteLine("TIMES 0x50000 db 0");
aOutputWriter.WriteLine("Kernel_Stack: ");
aOutputWriter.WriteLine("");
aOutputWriter.WriteLine("");
aOutputWriter.WriteLine("_end: ; end of BSS - here's the virtual and logical end.");
}
}
protected override void EmitFooter(string aGroup, StreamWriter aOutputWriter) {
if (aGroup == MainGroup) {
aOutputWriter.WriteLine("_end_data: ; -- end of CODE+DATA ");
}
base.EmitFooter(aGroup, aOutputWriter);
}
protected override void EmitHeader(string aGroup, StreamWriter aOutputWriter) {
base.EmitHeader(aGroup, aOutputWriter);
//mOutputWriter.WriteLine("format ms coff ");
//mOutputWriter.WriteLine("org 0220000h ; the best place to load our kernel to. ");
//mOutputWriter.WriteLine("use32 ; the kernel will be run in 32-bit protected mode, ");
aOutputWriter.WriteLine("");
// List<string> xTheLabels = new List<string>();
// string mLastRealLabel = "";
// foreach (IL2CPU.Assembler.Instruction x in mInstructions) {
// Label xLabel = x as Label;
// if (xLabel != null) {
// if (!xLabel.Name.StartsWith(".")) {
// mLastRealLabel = xLabel.Name;
// xTheLabels.Add(xLabel.Name);
// } else {
// xTheLabels.Add(mLastRealLabel + "@@" + xLabel.Name.Substring(1));
// }
// }
// }
// if (xTheLabels.Count > 0) {
// mOutputWriter.WriteLine("include 'export.inc'");
// mOutputWriter.WriteLine("section '.edata' export data readable");
// mOutputWriter.WriteLine("");
// mOutputWriter.WriteLine("\texport 'OUTPUT.dll', \\");
// for (int i = 0; i < xTheLabels.Count; i++) {
// if (i == (xTheLabels.Count - 1)) {
// mOutputWriter.WriteLine("\t\t {0},'{1}'", xTheLabels[i].Replace("@@", "."), xTheLabels[i]);
// } else {
// mOutputWriter.WriteLine("\t\t{0},'{1}', \\", xTheLabels[i].Replace("@@", "."), xTheLabels[i]);
// }
// public entry as 'entry'
//mOutputWriter.WriteLine("\tpublic {0} as '{0}'", xTheLabels[i].Replace("@@", "."), xTheLabels[i]);
// }
// }
//mOutputWriter.WriteLine("section '.code' code readable executable");
aOutputWriter.WriteLine("");
}
protected override void EmitImportMembers(string aGroup, StreamWriter aOutputWriter) {
if (ImportMembers.Count > 0) {
throw new Exception("You can't use external libraries in OS kernels");
}
}
}
}