mirror of
				https://github.com/jart/cosmopolitan.git
				synced 2025-10-25 10:40:57 +00:00 
			
		
		
		
	Initial import
This commit is contained in:
		
						commit
						c91b3c5006
					
				
					 14915 changed files with 590219 additions and 0 deletions
				
			
		
							
								
								
									
										42
									
								
								ape/lib/apelib.mk
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								ape/lib/apelib.mk
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,42 @@ | |||
| #-*-mode:makefile-gmake;indent-tabs-mode:t;tab-width:8;coding:utf-8-*-┐
 | ||||
| #───vi: set et ft=make ts=8 tw=8 fenc=utf-8 :vi───────────────────────┘
 | ||||
| 
 | ||||
| PKGS += APE_LIB | ||||
| 
 | ||||
| APE_LIB_ARTIFACTS += APE_LIB_A | ||||
| APE_LIB = $(APE_LIB_A_DEPS) $(APE_LIB_A) | ||||
| APE_LIB_A = o/$(MODE)/ape/lib/apelib.a | ||||
| APE_LIB_A_FILES := $(wildcard ape/lib/*) | ||||
| APE_LIB_A_HDRS = $(filter %.h,$(APE_LIB_A_FILES)) | ||||
| APE_LIB_A_SRCS_S = $(filter %.S,$(APE_LIB_A_FILES)) | ||||
| APE_LIB_A_SRCS_C = $(filter %.c,$(APE_LIB_A_FILES)) | ||||
| 
 | ||||
| APE_LIB_A_SRCS =					\
 | ||||
| 	$(APE_LIB_A_SRCS_S)				\
 | ||||
| 	$(APE_LIB_A_SRCS_C) | ||||
| 
 | ||||
| APE_LIB_A_OBJS =					\
 | ||||
| 	$(APE_LIB_A_SRCS_S:%.S=o/$(MODE)/%.o)		\
 | ||||
| 	$(APE_LIB_A_SRCS_C:%.c=o/$(MODE)/%.o)		\
 | ||||
| 	$(APE_LIB_A_SRCS:%=o/$(MODE)/%.zip.o)		\
 | ||||
| 	o/$(MODE)/NOTICE.zip.o | ||||
| 
 | ||||
| APE_LIB_A_CHECKS = $(APE_LIB_A_HDRS:%=o/$(MODE)/%.ok) | ||||
| APE_LIB_A_DIRECTDEPS = LIBC_STR LIBC_STUBS | ||||
| APE_LIB_A_DEPS = $(call uniq,$(foreach x,$(APE_LIB_A_DIRECTDEPS),$($(x)))) | ||||
| 
 | ||||
| $(APE_LIB_A): ape/lib/ $(APE_LIB_A).pkg $(APE_LIB_A_OBJS) | ||||
| $(APE_LIB_A).pkg: $(APE_LIB_A_OBJS) $(foreach x,$(APE_LIB_A_DIRECTDEPS),$($(x)_A).pkg) | ||||
| 
 | ||||
| APE_LIB_LIBS = $(foreach x,$(APE_LIB_ARTIFACTS),$($(x))) | ||||
| APE_LIB_SRCS = $(foreach x,$(APE_LIB_ARTIFACTS),$($(x)_SRCS)) | ||||
| APE_LIB_HDRS = $(foreach x,$(APE_LIB_ARTIFACTS),$($(x)_HDRS)) | ||||
| APE_LIB_BINS = $(foreach x,$(APE_LIB_ARTIFACTS),$($(x)_BINS)) | ||||
| APE_LIB_CHECKS = $(foreach x,$(APE_LIB_ARTIFACTS),$($(x)_CHECKS)) | ||||
| APE_LIB_OBJS = $(foreach x,$(APE_LIB_ARTIFACTS),$($(x)_OBJS)) | ||||
| $(APE_LIB_OBJS): $(BUILD_FILES) libc/str/str.mk | ||||
| 
 | ||||
| .PHONY: o/$(MODE)/ape/lib | ||||
| o/$(MODE)/ape/lib:			\ | ||||
| 		$(APE_LIB_CHECKS)	\
 | ||||
| 		$(APE_LIB_A) | ||||
							
								
								
									
										59
									
								
								ape/lib/apm.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								ape/lib/apm.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,59 @@ | |||
| /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | ||||
| │vi: set net ft=c ts=8 sts=2 sw=2 fenc=utf-8                                :vi│ | ||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||
| │                                                                              │ | ||||
| │ 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; version 2 of the License.                      │ | ||||
| │                                                                              │ | ||||
| │ 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., 51 Franklin Street, Fifth Floor, Boston, MA                │ | ||||
| │ 02110-1301 USA                                                               │ | ||||
| ╠──────────────────────────────────────────────────────────────────────────────╣ | ||||
| │░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░█▀█░█▀█░▀█▀░█░█░█▀█░█░░░█░░░█░█░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░█▀█░█░▄░░█░░█░█░█▀█░█░░░█░░░▀█▀░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░▀░▀░▀▀▀░░▀░░▀▀▀░▀░▀░▀▀▀░▀▀▀░░▀░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░█▀█░█▀█░█▀█░▀█▀░█▀█░█▀█░█░░░█▀▀░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░█▀▀░█ █░██▀░░█░░█▀█░█▀█░█░░░█▀▀░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░▀░░░▀▀▀░▀░▀░░▀░░▀░▀░▀▀▀░▀▀▀░▀▀▀░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░█▀▀░█░█░█▀▀░█▀█░█░█░▀█▀░█▀█░█▀█░█░░█▀▀░░░░░░░░░░░░░░░░░░░░░░░░▄▄░░░▐█░░│ | ||||
| │░░░░░░░█▀▀░▄▀▄░█▀▀░█░▄░█░█░░█░░█▀█░█▀█░█░░█▀▀░░░░░░░░░░░░▄▄▄░░░▄██▄░░█▀░░░█░▄░│ | ||||
| │░░░░░░░▀▀▀░▀░▀░▀▀▀░▀▀▀░▀▀▀░░▀░░▀░▀░▀▀▀░▀▀░▀▀▀░░░░░░░░░░▄██▀█▌░██▄▄░░▐█▀▄░▐█▀░░│ | ||||
| │░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░▐█▀▀▌░░░▄▀▌░▌░█░▌░░▌░▌░░│ | ||||
| ╠──────────────────────────────────────────────────────▌▀▄─▐──▀▄─▐▄─▐▄▐▄─▐▄─▐▄─│ | ||||
| │ αcτµαlly pδrταblε εxεcµταblε § green energy                                  │ | ||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||
| #ifndef APE_LIB_APM_H_ | ||||
| #define APE_LIB_APM_H_ | ||||
| 
 | ||||
| /**
 | ||||
|  * @fileoverview Advanced Power Management. | ||||
|  * | ||||
|  * <p>APM is useful for exiting programs, without needing to ask the | ||||
|  * human to flip a physical switch or pass QEMU's -no-reboot flag. | ||||
|  * | ||||
|  * <p><b>Implementation Detail:</b> Supporting ACPI would literally | ||||
|  * require implementing a programming language. | ||||
|  * | ||||
|  * @see APM BIOS Interface Specification v1.2 | ||||
|  * @since IBM PC/AT | ||||
|  */ | ||||
| 
 | ||||
| #define APM_SERVICE 0x15 | ||||
| 
 | ||||
| #if !(__ASSEMBLER__ + __LINKER__ + 0) | ||||
| 
 | ||||
| void apmoff(void) noreturn; | ||||
| 
 | ||||
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | ||||
| #endif /* APE_LIB_APM_H_ */ | ||||
							
								
								
									
										36
									
								
								ape/lib/bootdr.S
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								ape/lib/bootdr.S
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,36 @@ | |||
| /*-*- mode:asm; indent-tabs-mode:t; tab-width:8; coding:utf-8               -*-│
 | ||||
| │vi: set et ft=asm ts=8 sw=8 fenc=utf-8                                     :vi│ | ||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||
| │                                                                              │ | ||||
| │ 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; version 2 of the License.                      │
 | ||||
| │                                                                              │ | ||||
| │ 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., 51 Franklin Street, Fifth Floor, Boston, MA                │ | ||||
| │ 02110-1301 USA                                                               │ | ||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||
| #include "ape/macros.h" | ||||
| #include "ape/notice.inc" | ||||
| .section .real,"ax",@progbits
 | ||||
| .yoink	__FILE__
 | ||||
| .code16 | ||||
| 
 | ||||
| /	Resets personal computer. | ||||
| / | ||||
| /	@param	di drive number, e.g. A:\ is 0x00, C:\ is 0x80
 | ||||
| /	@mode	real
 | ||||
| /	@noreturn
 | ||||
| bootdr:	push	%bp | ||||
| 	mov	%sp,%bp | ||||
| 	mov	%di,%dx | ||||
| 	int	$0x19 | ||||
| 	ljmp	$0xf000,$0xfff0 | ||||
| 	.endfn	bootdr,globl | ||||
							
								
								
									
										38
									
								
								ape/lib/e820map.S
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								ape/lib/e820map.S
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,38 @@ | |||
| /*-*- mode:asm; indent-tabs-mode:t; tab-width:8; coding:utf-8               -*-│
 | ||||
| │vi: set et ft=asm ts=8 tw=8 fenc=utf-8                                     :vi│ | ||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||
| │                                                                              │ | ||||
| │ 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; version 2 of the License.                      │
 | ||||
| │                                                                              │ | ||||
| │ 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., 51 Franklin Street, Fifth Floor, Boston, MA                │ | ||||
| │ 02110-1301 USA                                                               │ | ||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||
| #include "ape/lib/pc.h" | ||||
| #include "ape/config.h" | ||||
| #include "ape/macros.h" | ||||
| #include "ape/notice.inc" | ||||
| .section .real,"ax",@progbits
 | ||||
| .yoink	__FILE__
 | ||||
| .code16 | ||||
| 
 | ||||
| 	.globl	e820map
 | ||||
| 	.hidden	e820map
 | ||||
| 	.type	e820map,@object
 | ||||
| 	.size	e820map,XLM_E820_SIZE | ||||
| 	e820map = ape.xlm + XLM_E820 | ||||
| 
 | ||||
| 	.globl	e820map_xlm
 | ||||
| 	.hidden	e820map_xlm
 | ||||
| 	.type	e820map_xlm,@object
 | ||||
| 	.size	e820map_xlm,XLM_E820_SIZE | ||||
| 	e820map_xlm = XLM(E820) | ||||
							
								
								
									
										59
									
								
								ape/lib/flattenhighmemory.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								ape/lib/flattenhighmemory.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,59 @@ | |||
| /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | ||||
| │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8                                :vi│ | ||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||
| │                                                                              │ | ||||
| │ 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; version 2 of the License.                      │ | ||||
| │                                                                              │ | ||||
| │ 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., 51 Franklin Street, Fifth Floor, Boston, MA                │ | ||||
| │ 02110-1301 USA                                                               │ | ||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||
| #include "ape/config.h" | ||||
| #include "ape/lib/pc.h" | ||||
| #include "libc/bits/bits.h" | ||||
| #include "libc/bits/safemacros.h" | ||||
| 
 | ||||
| /**
 | ||||
|  * Virtualizes physical memory. | ||||
|  * | ||||
|  * This function removes memory holes (discovered by e820() earlier) and | ||||
|  * creates the illusion of flat contiguous memory for as much RAM as the | ||||
|  * BIOS reports usable. Memory is safe to use and remap afterwards. | ||||
|  * | ||||
|  * @see ape/ape.S | ||||
|  */ | ||||
| textreal void flattenhighmemory(struct SmapEntry *e820, struct PageTable *pml4t, | ||||
|                                 uint64_t *ptsp) { | ||||
|   struct SmapEntry *smap = e820; | ||||
|   struct SmapEntry *hole = e820; | ||||
|   uint64_t paddr = IMAGE_BASE_PHYSICAL; | ||||
|   uint64_t vaddr = IMAGE_BASE_VIRTUAL; | ||||
|   while (smap->size) { | ||||
|     while (smap->size && smap->type != kMemoryUsable) smap++; | ||||
|     paddr = roundup(max(paddr, smap->addr), PAGESIZE); | ||||
|     while (paddr < rounddown(smap->addr + smap->size, PAGESIZE)) { | ||||
|       while (hole->size && | ||||
|              (hole->type == kMemoryUsable || hole->addr + hole->size < paddr)) { | ||||
|         hole++; | ||||
|       } | ||||
|       if (paddr >= hole->addr && paddr < hole->addr + hole->size) { | ||||
|         paddr = roundup(hole->addr + hole->size, PAGESIZE); | ||||
|       } else { | ||||
|         uint64_t *entry = getpagetableentry(vaddr, 3, pml4t, ptsp); | ||||
|         *entry = paddr | PAGE_V | PAGE_RW; | ||||
|         vaddr += 0x1000; | ||||
|         paddr += 0x1000; | ||||
|       } | ||||
|     } | ||||
|     smap++; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										32
									
								
								ape/lib/g_pml4t.S
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								ape/lib/g_pml4t.S
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,32 @@ | |||
| /*-*- mode:asm; indent-tabs-mode:t; tab-width:8; coding:utf-8               -*-│
 | ||||
| │vi: set et ft=asm ts=8 tw=8 fenc=utf-8                                     :vi│ | ||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||
| │                                                                              │ | ||||
| │ 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; version 2 of the License.                      │
 | ||||
| │                                                                              │ | ||||
| │ 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., 51 Franklin Street, Fifth Floor, Boston, MA                │ | ||||
| │ 02110-1301 USA                                                               │ | ||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||
| #include "ape/lib/pc.h" | ||||
| #include "ape/config.h" | ||||
| #include "ape/macros.h" | ||||
| #include "ape/notice.inc" | ||||
| .section .real,"ax",@progbits
 | ||||
| .yoink	__FILE__
 | ||||
| .code16 | ||||
| 
 | ||||
| 	.globl	g_pml4t
 | ||||
| 	.hidden	g_pml4t
 | ||||
| 	.type	g_pml4t,@object
 | ||||
| 	.size	g_pml4t,0x1000 | ||||
| 	g_pml4t = REAL_STACK_FRAME - 0x1000 | ||||
							
								
								
									
										38
									
								
								ape/lib/g_ptsp.S
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								ape/lib/g_ptsp.S
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,38 @@ | |||
| /*-*- mode:asm; indent-tabs-mode:t; tab-width:8; coding:utf-8               -*-│
 | ||||
| │vi: set et ft=asm ts=8 tw=8 fenc=utf-8                                     :vi│ | ||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||
| │                                                                              │ | ||||
| │ 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; version 2 of the License.                      │
 | ||||
| │                                                                              │ | ||||
| │ 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., 51 Franklin Street, Fifth Floor, Boston, MA                │ | ||||
| │ 02110-1301 USA                                                               │ | ||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||
| #include "ape/lib/pc.h" | ||||
| #include "ape/config.h" | ||||
| #include "ape/macros.h" | ||||
| #include "ape/notice.inc" | ||||
| .section .real,"ax",@progbits
 | ||||
| .yoink	__FILE__
 | ||||
| .code16 | ||||
| 
 | ||||
| 	.globl	g_ptsp
 | ||||
| 	.hidden	g_ptsp
 | ||||
| 	.type	g_ptsp,@object
 | ||||
| 	.size	g_ptsp,XLM_PAGE_TABLE_STACK_POINTER_SIZE | ||||
| 	g_ptsp = ape.xlm + XLM_PAGE_TABLE_STACK_POINTER | ||||
| 
 | ||||
| 	.globl	g_ptsp_xlm
 | ||||
| 	.hidden	g_ptsp_xlm
 | ||||
| 	.type	g_ptsp_xlm,@object
 | ||||
| 	.size	g_ptsp_xlm,XLM_PAGE_TABLE_STACK_POINTER_SIZE | ||||
| 	g_ptsp_xlm = XLM(PAGE_TABLE_STACK_POINTER) | ||||
							
								
								
									
										40
									
								
								ape/lib/getpagetableentry.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								ape/lib/getpagetableentry.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,40 @@ | |||
| /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | ||||
| │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8                                :vi│ | ||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||
| │                                                                              │ | ||||
| │ 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; version 2 of the License.                      │ | ||||
| │                                                                              │ | ||||
| │ 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., 51 Franklin Street, Fifth Floor, Boston, MA                │ | ||||
| │ 02110-1301 USA                                                               │ | ||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||
| #include "ape/lib/pc.h" | ||||
| #include "libc/assert.h" | ||||
| 
 | ||||
| textreal static uint64_t pushpagetable(uint64_t *ptsp) { | ||||
|   return (*ptsp -= PAGESIZE) | PAGE_V | PAGE_RW; | ||||
| } | ||||
| 
 | ||||
| textreal uint64_t *getpagetableentry(uint64_t vaddr, unsigned depth, | ||||
|                                      struct PageTable *pml4t, uint64_t *ptsp) { | ||||
|   assert(depth <= 3); | ||||
|   assert(*ptsp % PAGESIZE == 0); | ||||
|   assert((intptr_t)pml4t % PAGESIZE == 0); | ||||
|   unsigned char shift = 39; | ||||
|   for (;;) { | ||||
|     uint64_t *entry = &pml4t->p[(vaddr >> shift) & 511]; | ||||
|     if (!depth--) return entry; | ||||
|     shift -= 9; | ||||
|     if (!*entry) *entry = pushpagetable(ptsp); | ||||
|     pml4t = (void *)(*entry & PAGE_TA); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										46
									
								
								ape/lib/kbiosdataarea.S
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								ape/lib/kbiosdataarea.S
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,46 @@ | |||
| /*-*- mode:asm; indent-tabs-mode:t; tab-width:8; coding:utf-8               -*-│
 | ||||
| │vi: set et ft=asm ts=8 tw=8 fenc=utf-8                                     :vi│ | ||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||
| │                                                                              │ | ||||
| │ 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; version 2 of the License.                      │
 | ||||
| │                                                                              │ | ||||
| │ 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., 51 Franklin Street, Fifth Floor, Boston, MA                │ | ||||
| │ 02110-1301 USA                                                               │ | ||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||
| #include "ape/lib/pc.h" | ||||
| #include "ape/config.h" | ||||
| #include "ape/macros.h" | ||||
| #include "ape/notice.inc" | ||||
| .section .real,"ax",@progbits
 | ||||
| .yoink	__FILE__
 | ||||
| .code16 | ||||
| 
 | ||||
| 	.globl	kBiosDataArea
 | ||||
| 	.hidden	kBiosDataArea
 | ||||
| 	.type	kBiosDataArea,@object
 | ||||
| 	.size	kBiosDataArea,XLM_BIOS_DATA_AREA_SIZE | ||||
| 	kBiosDataArea = ape.xlm + XLM_BIOS_DATA_AREA | ||||
| 
 | ||||
| 	.globl	kBiosDataAreaXlm
 | ||||
| 	.hidden	kBiosDataAreaXlm
 | ||||
| 	.type	kBiosDataAreaXlm,@object
 | ||||
| 	.size	kBiosDataAreaXlm,XLM_BIOS_DATA_AREA_SIZE | ||||
| 	kBiosDataAreaXlm = XLM(BIOS_DATA_AREA) | ||||
| 
 | ||||
| 	.section .sort.real.init.2.kBiosDataArea,"ax",@progbits
 | ||||
| 	movpp	%ds,%es			# copy bios data to valid page | ||||
| 	mov	$PC_BIOS_DATA_AREA,%si | ||||
| 	mov	$XLM(BIOS_DATA_AREA),%di | ||||
| 	mov	$XLM_BIOS_DATA_AREA_SIZE,%cx | ||||
| 	rep movsb | ||||
| 	.previous | ||||
							
								
								
									
										27
									
								
								ape/lib/pageunmap.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								ape/lib/pageunmap.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | |||
| /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | ||||
| │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8                                :vi│ | ||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||
| │                                                                              │ | ||||
| │ 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; version 2 of the License.                      │ | ||||
| │                                                                              │ | ||||
| │ 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., 51 Franklin Street, Fifth Floor, Boston, MA                │ | ||||
| │ 02110-1301 USA                                                               │ | ||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||
| #include "ape/lib/pc.h" | ||||
| #include "libc/bits/bits.h" | ||||
| 
 | ||||
| textreal void pageunmap(uint64_t vaddr) { | ||||
|   uint64_t *entry = getpagetableentry(vaddr, 3, &g_pml4t, &g_ptsp_xlm); | ||||
|   *entry &= ~PAGE_V; | ||||
|   invlpg(vaddr); | ||||
| } | ||||
							
								
								
									
										239
									
								
								ape/lib/pc.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										239
									
								
								ape/lib/pc.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,239 @@ | |||
| /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | ||||
| │vi: set net ft=c ts=8 sts=2 sw=2 fenc=utf-8                                :vi│ | ||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||
| │                                                                              │ | ||||
| │ 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; version 2 of the License.                      │ | ||||
| │                                                                              │ | ||||
| │ 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., 51 Franklin Street, Fifth Floor, Boston, MA                │ | ||||
| │ 02110-1301 USA                                                               │ | ||||
| ╠──────────────────────────────────────────────────────────────────────────────╣ | ||||
| │░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░█▀█░█▀█░▀█▀░█░█░█▀█░█░░░█░░░█░█░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░█▀█░█░▄░░█░░█░█░█▀█░█░░░█░░░▀█▀░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░▀░▀░▀▀▀░░▀░░▀▀▀░▀░▀░▀▀▀░▀▀▀░░▀░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░█▀█░█▀█░█▀█░▀█▀░█▀█░█▀█░█░░░█▀▀░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░█▀▀░█ █░██▀░░█░░█▀█░█▀█░█░░░█▀▀░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░▀░░░▀▀▀░▀░▀░░▀░░▀░▀░▀▀▀░▀▀▀░▀▀▀░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░│ | ||||
| │░░░░░░░█▀▀░█░█░█▀▀░█▀█░█░█░▀█▀░█▀█░█▀█░█░░█▀▀░░░░░░░░░░░░░░░░░░░░░░░░▄▄░░░▐█░░│ | ||||
| │░░░░░░░█▀▀░▄▀▄░█▀▀░█░▄░█░█░░█░░█▀█░█▀█░█░░█▀▀░░░░░░░░░░░░▄▄▄░░░▄██▄░░█▀░░░█░▄░│ | ||||
| │░░░░░░░▀▀▀░▀░▀░▀▀▀░▀▀▀░▀▀▀░░▀░░▀░▀░▀▀▀░▀▀░▀▀▀░░░░░░░░░░▄██▀█▌░██▄▄░░▐█▀▄░▐█▀░░│ | ||||
| │░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░░▐█▀▀▌░░░▄▀▌░▌░█░▌░░▌░▌░░│ | ||||
| ╠──────────────────────────────────────────────────────▌▀▄─▐──▀▄─▐▄─▐▄▐▄─▐▄─▐▄─│ | ||||
| │ αcτµαlly pδrταblε εxεcµταblε § ibm personal computer                         │ | ||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||
| #ifndef APE_LIB_PC_H_ | ||||
| #define APE_LIB_PC_H_ | ||||
| 
 | ||||
| #define BOOTSIG           0xaa55 /* master boot record signature */ | ||||
| #define PC_BIOS_DATA_AREA 0x400 | ||||
| 
 | ||||
| #define kInterruptFlag (1u << 9) | ||||
| 
 | ||||
| /* FPU Status Word (x87)
 | ||||
|    @see Intel Manual V1 §8.1.3 | ||||
|    IE: Invalid Operation ────────────────┐ | ||||
|    DE: Denormalized Operand ────────────┐│ | ||||
|    ZE: Zero Divide ────────────────────┐││ | ||||
|    OE: Overflow Flag ─────────────────┐│││ | ||||
|    UE: Underflow Flag ───────────────┐││││ | ||||
|    PE: Precision Flag ──────────────┐│││││ | ||||
|    SF: Stack Fault ────────────────┐││││││ | ||||
|    ES: Exception Summary Status ──┐│││││││ | ||||
|    C0-3: Condition Codes ──┬────┐ ││││││││ | ||||
|    Top of Stack Pointer ─────┐  │ ││││││││ | ||||
|    B: FPU Busy ───────────┐│ │  │ ││││││││ | ||||
|                           ││┌┴┐┌┼┐││││││││ | ||||
|                           │↓│ │↓↓↓││││││││*/ | ||||
| #define FPU_ZE 0b0000000000100000000000100 | ||||
| #define FPU_C0 0b0000000000000000100000000 | ||||
| #define FPU_C1 0b0000000000000001000000000 | ||||
| #define FPU_C2 0b0000000000000010000000000 | ||||
| #define FPU_C3 0b0000000000100000000000000 | ||||
| 
 | ||||
| #define CR0_PE (1u << 0)  /* protected mode enabled */ | ||||
| #define CR0_MP (1u << 1)  /* monitor coprocessor */ | ||||
| #define CR0_EM (1u << 2)  /* no x87 fpu present if set */ | ||||
| #define CR0_TS (1u << 3)  /* task switched x87 */ | ||||
| #define CR0_ET (1u << 4)  /* extension type 287 or 387 */ | ||||
| #define CR0_NE (1u << 5)  /* enable x87 error reporting */ | ||||
| #define CR0_WP (1u << 16) /* write protect read-only pages @pl0 */ | ||||
| #define CR0_AM (1u << 18) /* alignment mask */ | ||||
| #define CR0_NW (1u << 29) /* global write-through cache disable */ | ||||
| #define CR0_CD (1u << 30) /* global cache disable */ | ||||
| #define CR0_PG (1u << 31) /* paging enabled */ | ||||
| 
 | ||||
| #define CR4_VME        (1u << 0)  /* virtual 8086 mode extension */ | ||||
| #define CR4_PVI        (1u << 1)  /* protected mode virtual interrupts */ | ||||
| #define CR4_TSD        (1u << 2)  /* time stamp disable (rdtsc) */ | ||||
| #define CR4_DE         (1u << 3)  /* debugging extensions */ | ||||
| #define CR4_PSE        (1u << 4)  /* page size extension */ | ||||
| #define CR4_PAE        (1u << 5)  /* physical address extension */ | ||||
| #define CR4_MCE        (1u << 6)  /* machine check exception */ | ||||
| #define CR4_PGE        (1u << 7)  /* page global enabled */ | ||||
| #define CR4_OSFXSR     (1u << 9)  /* enable SSE and fxsave/fxrestor */ | ||||
| #define CR4_OSXMMEXCPT (1u << 10) /* enable unmasked SSE exceptions */ | ||||
| #define CR4_LA57       (1u << 12) /* enable level-5 paging */ | ||||
| #define CR4_VMXE       (1u << 13) /* enable VMX operations */ | ||||
| #define CR4_SMXE       (1u << 14) /* enable SMX operations */ | ||||
| #define CR4_FSGSBASE   (1u << 16) /* enable *FSBASE and *GSBASE instructions */ | ||||
| #define CR4_PCIDE      (1u << 17) /* enable process-context identifiers */ | ||||
| #define CR4_OSXSAVE    (1u << 18) /* enable XSAVE */ | ||||
| 
 | ||||
| #define XCR0_X87       (1u << 0) | ||||
| #define XCR0_SSE       (1u << 1) | ||||
| #define XCR0_AVX       (1u << 2) | ||||
| #define XCR0_BNDREG    (1u << 3) | ||||
| #define XCR0_BNDCSR    (1u << 4) | ||||
| #define XCR0_OPMASK    (1u << 5) | ||||
| #define XCR0_ZMM_HI256 (1u << 6) | ||||
| #define XCR0_HI16_ZMM  (1u << 7) | ||||
| 
 | ||||
| #define EFER     0xC0000080 /* extended feature enable register */ | ||||
| #define EFER_SCE (1u << 0)  /* system call extensions */ | ||||
| #define EFER_LME (1u << 8)  /* long mode enable */ | ||||
| #define EFER_LMA (1u << 10) /* long mode active */ | ||||
| #define EFER_NXE (1u << 11) /* no-execute enable */ | ||||
| 
 | ||||
| #define GDT_REAL_CODE   8 | ||||
| #define GDT_REAL_DATA   16 | ||||
| #define GDT_LEGACY_CODE 24 | ||||
| #define GDT_LEGACY_DATA 32 | ||||
| #define GDT_LONG_CODE   40 | ||||
| #define GDT_LONG_DATA   48 | ||||
| 
 | ||||
| #define PIC1         0x20 /* IO base address for master PIC */ | ||||
| #define PIC2         0xA0 /* IO base address for slave PIC */ | ||||
| #define PIC1_CMD     PIC1 | ||||
| #define PIC1_DATA    (PIC1 + 1) | ||||
| #define PIC2_CMD     PIC2 | ||||
| #define PIC2_DATA    (PIC2 + 1) | ||||
| #define PIC_EOI      0x20 /* End-of-interrupt command code */ | ||||
| #define PIC_READ_IRR 0x0a /* OCW3 irq ready next CMD read */ | ||||
| #define PIC_READ_ISR 0x0b /* OCW3 irq service next CMD read */ | ||||
| 
 | ||||
| /* Long Mode Paging
 | ||||
|    @see Intel Manual V.3A §4.1 §4.5 | ||||
|                                           IsValid (ignored on CR3) V┐ | ||||
|      ┌Block Instr. Fetches (if NXE)  IsWritable (ignored on CR3) RW┐│ | ||||
|      │                                 Permit User-Mode Access - u┐││ | ||||
|      │                             Page-level Write-Through - PWT┐│││ | ||||
|      │                            Page-level Cache Disable - PCD┐││││ | ||||
|      │                          Set if has been read - Accessed┐│││││ | ||||
|      │                         Set if has been written - Dirty┐││││││ | ||||
|      │                   IsPage (if PDPTE/PDE) or PAT (if PT)┐│││││││ | ||||
|      │        (If this maps 2MB/1GB page and CR4.PGE) Global┐││││││││ | ||||
|      │      (If IsPage 2MB/1GB, see Intel V3A § 11.12) PAT  │││││││││ | ||||
|      │                                                  │   │││││││││ | ||||
|      │             ┌────────────────────────────────────┤   │││││││││ | ||||
|      │   Must Be 0┐│ Next Page Table Address (!IsPage)  │   │││││││││ | ||||
|      │            │├────────────────────────────────────┤   │││││││││ | ||||
|      │            ││ Physical Address 4KB               │   │││││││││ | ||||
|      │┌───┐┌─────┐│├───────────────────────────┐        │ign│││││││││ | ||||
|      ││PKE││ ign │││ Physical Address 2MB      │        │┌┴┐│││││││││ | ||||
|      ││   ││     ││├──────────────────┐        │        ││ ││││││││││ | ||||
|      ││   ││     │││ Phys. Addr. 1GB  │        │        ││ ││││││││││ | ||||
|      ││   ││     │││                  │        │        ││ ││││││││││ | ||||
|      0b00000000000011111111111111111111111111111111111111000000000000 | ||||
|      6666555555555544444444443333333333222222222211111111110000000000 | ||||
|      3210987654321098765432109876543210987654321098765432109876543210*/ | ||||
| #define PAGE_V   /*                                      */ 0b000000001 | ||||
| #define PAGE_RW  /*                                     */ 0b000000010 | ||||
| #define PAGE_U   /*                                      */ 0b000000100 | ||||
| #define PAGE_2MB /*                                    */ 0b110000000 | ||||
| #define PAGE_1GB /*                                    */ 0b110000000 | ||||
| #define PAGE_TA  0b011111111111111111111111111111111111111000000000000 | ||||
| #define PAGE_PA2 0b11111111111111111111111111111000000000000000000000 | ||||
| 
 | ||||
| #if !(__ASSEMBLER__ + __LINKER__ + 0) | ||||
| #include "ape/config.h" | ||||
| 
 | ||||
| struct thatispacked GlobalDescriptorTable { | ||||
|   uint16_t size; | ||||
|   uint64_t *entries; | ||||
| }; | ||||
| 
 | ||||
| /**
 | ||||
|  * Memory hole map. | ||||
|  * @see wiki.osdev.org/Detecting_Memory_(x86) | ||||
|  * @since 2002 | ||||
|  */ | ||||
| struct SmapEntry { | ||||
|   uint64_t addr; | ||||
|   uint64_t size; | ||||
|   enum { | ||||
|     kMemoryUsable = 1, | ||||
|     kMemoryUnusable = 2, | ||||
|     kMemoryAcpiReclaimable = 3, | ||||
|     kMemoryAcpiNvs = 4, | ||||
|     kMemoryBad = 5 | ||||
|   } type; | ||||
|   uint32_t __acpi3; /* is abstracted */ | ||||
| }; | ||||
| 
 | ||||
| struct IdtDescriptor { | ||||
|   uint16_t offset_1; /* offset bits 0..15 */ | ||||
|   uint16_t selector; /* a code segment selector in GDT or LDT */ | ||||
|   uint8_t ist;       /* bits 0..2 hold stack table offset, rest are zero */ | ||||
|   uint8_t type_attr; /* type and attributes */ | ||||
|   uint16_t offset_2; /* offset bits 16..31 */ | ||||
|   uint32_t offset_3; /* offset bits 32..63 */ | ||||
|   uint32_t zero;     /* reserved */ | ||||
| }; | ||||
| 
 | ||||
| struct thatispacked PageTable { | ||||
|   uint64_t p[512]; | ||||
| } aligned(PAGESIZE); | ||||
| 
 | ||||
| extern struct PageTable g_pml4t; | ||||
| extern struct GlobalDescriptorTable gdt; | ||||
| 
 | ||||
| extern const unsigned char kBiosDataArea[256]; | ||||
| extern const unsigned char kBiosDataAreaXlm[256]; | ||||
| 
 | ||||
| extern struct SmapEntry e820map[XLM_E820_SIZE / sizeof(struct SmapEntry)]; | ||||
| extern struct SmapEntry e820map_xlm[XLM_E820_SIZE / sizeof(struct SmapEntry)]; | ||||
| 
 | ||||
| extern uint64_t g_ptsp; | ||||
| extern uint64_t g_ptsp_xlm; | ||||
| 
 | ||||
| void bootdr(char drive) noreturn; | ||||
| 
 | ||||
| void smapsort(struct SmapEntry *); | ||||
| uint64_t *getpagetableentry(uint64_t, unsigned, struct PageTable *, uint64_t *); | ||||
| void flattenhighmemory(struct SmapEntry *, struct PageTable *, uint64_t *); | ||||
| void pageunmap(uint64_t); | ||||
| 
 | ||||
| forceinline unsigned long eflags(void) { | ||||
|   unsigned long res; | ||||
|   asm("pushf\n\t" | ||||
|       "pop\t%0" | ||||
|       : "=rm"(res)); | ||||
|   return res; | ||||
| } | ||||
| 
 | ||||
| forceinline unsigned char inb(unsigned short port) { | ||||
|   unsigned char al; | ||||
|   asm volatile("inb\t%1,%0" : "=a"(al) : "dN"(port)); | ||||
|   return al; | ||||
| } | ||||
| 
 | ||||
| forceinline void outb(unsigned short port, unsigned char byte) { | ||||
|   asm volatile("outb\t%0,%1" | ||||
|                : /* no inputs */ | ||||
|                : "a"(byte), "dN"(port)); | ||||
| } | ||||
| 
 | ||||
| #endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */ | ||||
| #endif /* APE_LIB_PC_H_ */ | ||||
							
								
								
									
										140
									
								
								ape/lib/pic.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										140
									
								
								ape/lib/pic.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,140 @@ | |||
| /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | ||||
| │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8                                :vi│ | ||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||
| │                                                                              │ | ||||
| │ 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; version 2 of the License.                      │ | ||||
| │                                                                              │ | ||||
| │ 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., 51 Franklin Street, Fifth Floor, Boston, MA                │ | ||||
| │ 02110-1301 USA                                                               │ | ||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||
| #include "ape/lib/pc.h" | ||||
| 
 | ||||
| #define ICW1_ICW4 0x01      /* ICW4 (not) needed */ | ||||
| #define ICW1_SINGLE 0x02    /* Single (cascade) mode */ | ||||
| #define ICW1_INTERVAL4 0x04 /* Call address interval 4 (8) */ | ||||
| #define ICW1_LEVEL 0x08     /* Level triggered (edge) mode */ | ||||
| #define ICW1_INIT 0x10      /* Initialization - required! */ | ||||
| 
 | ||||
| #define ICW4_8086 0x01       /* 8086/88 (MCS-80/85) mode */ | ||||
| #define ICW4_AUTO 0x02       /* Auto (normal) EOI */ | ||||
| #define ICW4_BUF_SLAVE 0x08  /* Buffered mode/slave */ | ||||
| #define ICW4_BUF_MASTER 0x0C /* Buffered mode/master */ | ||||
| #define ICW4_SFNM 0x10       /* Special fully nested (not) */ | ||||
| 
 | ||||
| static inline void io_wait(void) { | ||||
|   /* Magic technique from Linux, according to:
 | ||||
|    * wiki.osdev.org/index.php?title=Inline_Assembly/Examples&oldid=23541 | ||||
|    */ | ||||
|   outb(0x80, 0); | ||||
| } | ||||
| 
 | ||||
| void PIC_sendEOI(unsigned char irq) { | ||||
|   if (irq >= 8) outb(PIC2_CMD, PIC_EOI); | ||||
|   outb(PIC1_CMD, PIC_EOI); | ||||
| } | ||||
| 
 | ||||
| bool AreInterruptsEnabled() { | ||||
|   return (eflags() & kInterruptFlag) == kInterruptFlag; | ||||
| } | ||||
| 
 | ||||
| nodiscard forceinline unsigned long irqdisable(void) { | ||||
|   unsigned long eflags; | ||||
|   asm("pushf\n\t" | ||||
|       "cli\n\t" | ||||
|       "pop\t%0" | ||||
|       : "=r"(eflags) | ||||
|       : /* no inputs */ | ||||
|       : "cc"); | ||||
|   return eflags; | ||||
| } | ||||
| 
 | ||||
| forceinline void irqrestore(unsigned long eflags) { | ||||
|   asm volatile( | ||||
|       "push\t%0\n\t" | ||||
|       "popf" | ||||
|       : /* no outputs */ | ||||
|       : "rm"(eflags) | ||||
|       : "cc"); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * @param offset1 is vector offset for master PIC | ||||
|  *     vectors on the master become offset1..offset1+7 | ||||
|  * @param offset2 is same for slave PIC: offset2..offset2+7 | ||||
|  **/ | ||||
| void PIC_remap(int offset1, int offset2) { | ||||
|   unsigned char a1, a2; | ||||
|   a1 = inb(PIC1_DATA);  // save masks
 | ||||
|   a2 = inb(PIC2_DATA); | ||||
|   outb(PIC1_CMD, | ||||
|        ICW1_INIT | | ||||
|            ICW1_ICW4);  // starts the initialization sequence (in cascade mode)
 | ||||
|   io_wait(); | ||||
|   outb(PIC2_CMD, ICW1_INIT | ICW1_ICW4); | ||||
|   io_wait(); | ||||
|   outb(PIC1_DATA, offset1);  // ICW2: Master PIC vector offset
 | ||||
|   io_wait(); | ||||
|   outb(PIC2_DATA, offset2);  // ICW2: Slave PIC vector offset
 | ||||
|   io_wait(); | ||||
|   outb(PIC1_DATA, 4);  // ICW3: tell Master PIC that there is a slave PIC at
 | ||||
|                        // IRQ2 (0000 0100)
 | ||||
|   io_wait(); | ||||
|   outb(PIC2_DATA, 2);  // ICW3: tell Slave PIC its cascade identity (0000 0010)
 | ||||
|   io_wait(); | ||||
|   outb(PIC1_DATA, ICW4_8086); | ||||
|   io_wait(); | ||||
|   outb(PIC2_DATA, ICW4_8086); | ||||
|   io_wait(); | ||||
|   outb(PIC1_DATA, a1);  // restore saved masks.
 | ||||
|   outb(PIC2_DATA, a2); | ||||
| } | ||||
| 
 | ||||
| void IRQ_set_mask(unsigned char IRQline) { | ||||
|   uint16_t port; | ||||
|   uint8_t value; | ||||
|   if (IRQline < 8) { | ||||
|     port = PIC1_DATA; | ||||
|   } else { | ||||
|     port = PIC2_DATA; | ||||
|     IRQline -= 8; | ||||
|   } | ||||
|   value = inb(port) | (1 << IRQline); | ||||
|   outb(port, value); | ||||
| } | ||||
| 
 | ||||
| void IRQ_clear_mask(unsigned char IRQline) { | ||||
|   uint16_t port; | ||||
|   uint8_t value; | ||||
|   if (IRQline < 8) { | ||||
|     port = PIC1_DATA; | ||||
|   } else { | ||||
|     port = PIC2_DATA; | ||||
|     IRQline -= 8; | ||||
|   } | ||||
|   value = inb(port) & ~(1 << IRQline); | ||||
|   outb(port, value); | ||||
| } | ||||
| 
 | ||||
| static uint16_t __pic_get_irq_reg(int ocw3) { | ||||
|   /* OCW3 to PIC CMD to get the register values.  PIC2 is chained, and
 | ||||
|    * represents IRQs 8-15.  PIC1 is IRQs 0-7, with 2 being the chain */ | ||||
|   outb(PIC1_CMD, ocw3); | ||||
|   outb(PIC2_CMD, ocw3); | ||||
|   return (inb(PIC2_CMD) << 8) | inb(PIC1_CMD); | ||||
| } | ||||
| 
 | ||||
| /* Returns the combined value of the cascaded PICs irq request register */ | ||||
| uint16_t pic_get_irr(void) { return __pic_get_irq_reg(PIC_READ_IRR); } | ||||
| 
 | ||||
| /* Returns the combined value of the cascaded PICs in-service register */ | ||||
| uint16_t pic_get_isr(void) { return __pic_get_irq_reg(PIC_READ_ISR); } | ||||
							
								
								
									
										47
									
								
								ape/lib/smapsort.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								ape/lib/smapsort.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,47 @@ | |||
| /*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
 | ||||
| │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8                                :vi│ | ||||
| ╞══════════════════════════════════════════════════════════════════════════════╡ | ||||
| │ Copyright 2020 Justine Alexandra Roberts Tunney                              │ | ||||
| │                                                                              │ | ||||
| │ 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; version 2 of the License.                      │ | ||||
| │                                                                              │ | ||||
| │ 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., 51 Franklin Street, Fifth Floor, Boston, MA                │ | ||||
| │ 02110-1301 USA                                                               │ | ||||
| ╚─────────────────────────────────────────────────────────────────────────────*/ | ||||
| #include "ape/lib/pc.h" | ||||
| 
 | ||||
| textreal static unsigned smapcount(const struct SmapEntry *se) { | ||||
|   unsigned i = 0; | ||||
|   while (se[i].size) ++i; | ||||
|   return i; | ||||
| } | ||||
| 
 | ||||
| textreal static void smapsorter(size_t n, struct SmapEntry a[n]) { | ||||
|   struct SmapEntry t; | ||||
|   unsigned i, j; | ||||
|   for (i = 1; i < n; ++i) { | ||||
|     j = i; | ||||
|     t = a[i]; | ||||
|     while (j > 0 && (intptr_t)t.addr - (intptr_t)a[j - 1].addr) { | ||||
|       a[j] = a[j - 1]; | ||||
|       --j; | ||||
|     } | ||||
|     a[j] = t; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * Sorts BIOS e820 memory map. | ||||
|  */ | ||||
| textreal void smapsort(struct SmapEntry *smap) { | ||||
|   smapsorter(smapcount(smap), smap); | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue