diff --git a/data/Vortex86MX-eng/Makefile b/data/Vortex86MX-eng/Makefile index 03f143df0..55e69c70f 100644 --- a/data/Vortex86MX-eng/Makefile +++ b/data/Vortex86MX-eng/Makefile @@ -381,7 +381,7 @@ include Makefile.nasm include Makefile.copy # Special rules for copying sysfuncs.txt - it isn't directly included in the image. -docpack: $(DOCDIR)SYSFUNCS.TXT $(wildcard $(DOCDIR)*) +docpack: $(DOCDIR)SYSFUNCS.TXT $(DOCDIR)SYSFUNCS.TXT: $(KERNEL)/docs/sysfuncs.txt cp $(KERNEL)/docs/sysfuncs.txt $(DOCDIR)SYSFUNCS.TXT diff --git a/data/Vortex86MX-eng/Makefile.fasm b/data/Vortex86MX-eng/Makefile.fasm index 26e456d6e..67f4b4deb 100644 --- a/data/Vortex86MX-eng/Makefile.fasm +++ b/data/Vortex86MX-eng/Makefile.fasm @@ -28,25 +28,32 @@ # $(4) = name of program - without path and extension, define fasm_meta_rule $(1): $(2) Makefile.fasm .deps/.dir $$(call respace,$$(addsuffix .dir,$(3))) - fasm -m 65536 $$< "$$@" -s .deps/$(4).fas - prepsrc .deps/$(4).fas /dev/null - prepsrc .deps/$(4).fas /dev/stdout | \ - perl -n -e 's|\\|/|g;s| |\\ |g;push @a,$$$$1 if/^;include\\ \x27(.*?)\x27/;' \ - -e 'END{$$$$a=join " \\\n ",@a;print q`$(1): `,"$$$$a\n$$$$a:\n"}' > .deps/$(4).Po + tmpfile=`mktemp --tmpdir build.XXXXXXXX` && \ + (fasm -m 65536 "$$<" "$$@" -s $$$$tmpfile && \ + fasmdep -e $$$$tmpfile > .deps/$(4).Po && \ + rm $$$$tmpfile) || (rm $$$$tmpfile; false) kpack --nologo "$$@" -include .deps/$(4).Po endef +define fasm_nokpack_meta_rule +$(1): $(2) Makefile.fasm .deps/.dir $$(call respace,$$(addsuffix .dir,$(3))) + tmpfile=`mktemp --tmpdir build.XXXXXXXX` && \ + (fasm -m 65536 "$$<" "$$@" -s $$$$tmpfile && \ + fasmdep -e $$$$tmpfile > .deps/$(4).Po && \ + rm $$$$tmpfile) || (rm $$$$tmpfile; false) +-include .deps/$(4).Po +endef + progname=$(call respace,$(basename $(notdir $(call binarypart,$(f))))) binarydir=$(subst ./,,$(dir $(call binarypart,$(f)))) -$(foreach f,$(FASM_PROGRAMS) $(FASM_PROGRAMS_CD),$(eval $(call fasm_meta_rule,$(fbinary),$(fsource),$(binarydir),$(progname)))) +$(foreach f,$(FASM_PROGRAMS) $(FASM_PROGRAMS_CD) $(SKIN_SOURCES),$(eval $(call fasm_meta_rule,$(fbinary),$(fsource),$(binarydir),$(progname)))) +$(foreach f,$(FASM_NOKPACK_PROGRAMS),$(eval $(call fasm_nokpack_meta_rule,$(fbinary),$(fsource),$(binarydir),$(progname)))) # Rule for the kernel differs: it uses kerpack instead of kpack. kernel.mnt: $(KERNEL)/kernel.asm Makefile.fasm .deps/.dir - fasm -m 65536 $< "$@" -s .deps/kernel.fas - prepsrc .deps/kernel.fas /dev/null - prepsrc .deps/kernel.fas /dev/stdout | \ - perl -n -e 's|\\|/|g;s| |\\ |g;push @a,$$1 if/^;include\\ \x27(.*?)\x27/;' \ - -e 'END{$$a=join " \\\n ",@a;print "$@: $$a\n$$a:\n"}' > .deps/kernel.Po - kerpack $@ + tmpfile=`mktemp --tmpdir build.XXXXXXXX` && \ + (fasm -m 65536 "$<" "$@" -s $$tmpfile && \ + fasmdep -e $$tmpfile > .deps/kernel.Po && \ + rm $$tmpfile) || (rm $$tmpfile; false) -include .deps/kernel.Po diff --git a/data/common/fasmdep/fas.inc b/data/common/fasmdep/fas.inc new file mode 100644 index 000000000..0199b639f --- /dev/null +++ b/data/common/fasmdep/fas.inc @@ -0,0 +1,90 @@ +; Structures describing fasm symbol files. +define FAS_SIGNATURE 1A736166h +virtual at 0 +fas_header: +.signature dd ? ; Signature 1A736166h. +.major db ? ; Major version of flat assembler. +.minor db ? ; Minor version of flat assembler. +.headersize dw ? ; Length of header. +.input dd ? ; Offset of input file name in the strings table. +.output dd ? ; Offset of output file name in the strings table. +.strings_offs dd ? ; Offset of strings table. +.strings_size dd ? ; Length of strings table. +.symbols_offs dd ? ; Offset of symbols table. +.symbols_size dd ? ; Length of symbols table. +.preproc_offs dd ? ; Offset of preprocessed source. +.preproc_size dd ? ; Length of preprocessed source. +.asm_offs dd ? ; Offset of assembly dump. +.asm_size dd ? ; Length of assembly dump. +.section_offs dd ? ; Offset of section names table. +.section_size dd ? ; Length of section names table. +.symref_offs dd ? ; Offset of symbol references dump. +.symref_size dd ? ; Length of symbol references dump. +.size: +end virtual + +virtual at 0 +preproc_line_header: +; When the line was loaded from source, this field contains +; either zero (if it is the line from the main input file), or +; an offset inside the preprocessed source to the name of file, +; from which this line was loaded (the name of file is zero-ended +; string). +; When the line was generated by macroinstruction, this field +; contains offset inside the preprocessed source to the +; pascal-style string specifying the name of macroinstruction, +; which generated this line. +.source_name dd ? +; Bits 0-30 contain the number of this line. +; If the highest bit is zeroed, this line was loaded from source. +; If the highest bit is set, this line was generated by +; macroinstruction. +.line_number dd ? +; If the line was loaded from source, this field contains +; the position of the line inside the source file, from which it +; was loaded. +; If line was generated by macroinstruction, this field contains +; the offset of preprocessed line, which invoked the +; macroinstruction. If line was generated by instantaneous macro, +; this field is equal to the next one. +.line_offset dd ? +; If the line was generated by macroinstruction, this field +; contains offset of the preprocessed line inside the definition +; of macro, from which this one was generated. +.line_macro_offs dd ? +; The tokenized contents of line. +.contents: +end virtual + +virtual at 0 +asm_row: +.out_offs dd ? ; Offset in output file. +.preproc_offs dd ? ; Offset of line in preprocessed source. +.this dq ? ; Value of $ address. +.extended_sib dd ? ; Extended SIB for the $ address, the first two bytes + ; are register codes and the second two bytes are + ; corresponding scales. +; If the $ address is relocatable, this field contains +; information about section or external symbol, to which +; it is relative - otherwise this field is zero. +; When the highest bit is cleared, the address is +; relative to a section, and the bits 0-30 contain +; the index (starting from 1) in the table of sections. +; When the highest bit is set, the address is relative +; to an external symbol, and the bits 0-30 contain the +; the offset of the name of this symbol in the strings +; table. +.this_base dd ? +.this_type db ? ; Type of $ address value (as in table 2.2). +.code_type db ? ; Type of code - possible values are 16, 32, and 64. +; If the bit 0 is set, then at this point the assembly +; was taking place inside the virtual block, and the +; offset in output file has no meaning here. +; If the bit 1 is set, the line was assembled at the +; point, which was not included in the output file for +; some other reasons (like inside the reserved data at +; the end of section). +.flags db ? +.this_high db ? ; The higher bits of value of $ address. +.sizeof: +end virtual diff --git a/data/common/fasmdep/fasmdep.asm b/data/common/fasmdep/fasmdep.asm new file mode 100644 index 000000000..7a22db044 --- /dev/null +++ b/data/common/fasmdep/fasmdep.asm @@ -0,0 +1,765 @@ +; This program parses .fas file generated by fasm +; and prints the list of dependencies for make. +; Usage: fasmdep [-e] [ []]. +; If input file is not given, the program reads from stdin. +; If output file is not given, the program writes to stdout. +; If the option -e is given, the program also creates an empty +; goal for every dependency, this prevents make errors when +; files are renamed. + +; This definition controls the choice of platform-specific code. +;define OS WINDOWS +define OS LINUX + +; Program header. +match =WINDOWS,OS { include 'windows_header.inc' } +match =LINUX,OS { include 'linux_header.inc' } + +include 'fas.inc' + +; Main code +start: +; 1. Setup stack frame, zero-initialize all local variables. +virtual at ebp - .localsize +.begin: +.flags dd ? ; 1 if '-e' is given +.in dd ? ; handle of input file +.out dd ? ; handle of output file +.buf dd ? ; pointer to data of .fas file +.allocated dd ? ; number of bytes allocated for .buf +.free dd ? ; number of bytes free in .buf +.outstart dd ? ; offset in .buf for start of data to output +.names dd ? ; offset in .buf for start of output names +.include dd ? ; offset in .buf to value of %INCLUDE% +.testname dd ? ; offset in .buf for start of current file name +.prevfile dd ? ; offset in .buf for previous included file name +.prevfilefrom dd ? ; offset in .buf for .asm/.inc for .prevfile +.localsize = $ - .begin +match =LINUX,OS { +.argc dd ? +.argv: +} +end virtual + + mov ebp, esp + xor eax, eax +repeat .localsize / 4 + push eax +end repeat +; 2. Call the parser of the command line. + call get_params +; 3. Load the input file. +; Note that stdin can be a pipe, +; useful in bash-construction "fasm input.asm -s >(fasmdep > input.Po)", +; so its size is not always known in the beginning. +; So, we must read input file in portions, reallocating memory if needed. +.readloop: +; 3a. If the size is less than 32768 bytes, reallocate buffer. +; Otherwise, goto 3c. + cmp [.free], 32768 + jae .norealloc +; 3b. Reallocate buffer. +; Start with 65536 bytes and then double the size every time. + mov eax, 65536 + mov ecx, [.allocated] + add ecx, ecx + jz @f + mov eax, ecx +@@: + call realloc +.norealloc: +; 3c. Read the next portion. + call read + sub [.free], eax + test eax, eax + jnz .readloop +; 4. Sanity checks. +; We don't use .section_* and .symref_*, so allow them to be absent. + mov edi, [.allocated] + sub edi, [.free] +; Note that edi = number of bytes which were read +; and simultaneously pointer to free space in the buffer relative to [.buf]. + cmp edi, fas_header.section_offs + jb badfile + mov ebx, [.buf] + cmp [ebx+fas_header.signature], FAS_SIGNATURE + jnz badfile + cmp [ebx+fas_header.headersize], fas_header.section_offs + jb badfile +; 5. Get %INCLUDE% environment variable, it will be useful. + mov [.include], edi + mov esi, include_variable + sub esi, ebx + call get_environment_variable +; 6. Start writing dependencies: copy output and input files. + mov [.outstart], edi +; 6a. Copy output name. + mov esi, [ebx+fas_header.output] + call copy_asciiz_escaped +; 6b. Write ": ". + stdcall alloc_in_buf, 2 + mov word [edi+ebx], ': ' + inc edi + inc edi +; 6c. Copy input name. + mov [.names], edi + mov esi, [ebx+fas_header.input] + call copy_asciiz_escaped +; 7. Scan for 'include' dependencies. +; 7a. Get range for scanning. + mov edx, [ebx+fas_header.preproc_size] + mov esi, [ebx+fas_header.preproc_offs] + add edx, esi +.include_loop: +; 7b. Loop over preprocessed lines in the range. + cmp esi, edx + jae .include_done +; 7c. For every line, ignore header and do internal loop over tokens. + add esi, preproc_line_header.contents +.include_loop_int: +; There are five types of tokens: +; 1) "start preprocessor data" with code ';' +; 2) "quoted string" with code '"' +; 3) "word" with code 1Ah +; 4) one-byte tokens like "+", "(" and so on +; 5) "end-of-line" token with code 0. + mov al, [esi+ebx] + inc esi +; 7d. First, check token type parsing the first byte. +; For preprocessor tokens, continue to 7e. +; For quoted strings, go to 7g. +; For words, go to 7h. +; For end-of-line, exit the internal loop, continue the external loop. +; Otherwise, just continue the internal loop. + cmp al, ';' + jnz .notprep +; 7e. For "include" tokens length=7, token is "include", the next token is +; quoted string. +; These tokens are handled in 7f, other preprocessor tokens in 7g. + cmp byte [esi+ebx], 7 + jnz .notinclude + cmp dword [esi+ebx+1], 'incl' + jnz .notinclude + cmp dword [esi+ebx+5], 'ude"' + jnz .notinclude +; 7f. Skip over end of this token and the type byte of the next token; +; this gives the pointer to length-prefixed string, which should be added +; to dependencies. Note that doing that skips that string, +; so after copying just continue the internal loop. + add esi, 9 + call add_separator + call copy_name_escaped + jmp .include_loop_int +.quoted: +; 7g. To skip a word, load the dword length and add it to pointer. + add esi, [esi+ebx] + add esi, 4 + jmp .include_loop_int +.notinclude: +; 7h. To skip this token, load the byte length and add it to pointer. +; Note that word tokens and preprocessor tokens have a similar structure, +; so both are handled here. + movzx eax, byte [esi+ebx] + lea esi, [esi+eax+1] + jmp .include_loop_int +.notprep: + cmp al, 1Ah + jz .notinclude + cmp al, '"' + jz .quoted + test al, al + jnz .include_loop_int + jmp .include_loop +.include_done: +; 8. Scan for 'file' dependencies. +; 8a. Get range for scanning. +; Note that fas_header.asm_size can be slighly greater than the +; real size, so we break from the loop when there is no space left +; for the entire asm_row structure, not when .asm_size is exactly reached. + mov esi, [ebx+fas_header.asm_offs] + mov edx, [ebx+fas_header.asm_size] + add edx, esi + sub edx, asm_row.sizeof + push edx + jc .file_done +.file_loop: +; 8b. Loop over assembled lines in the range. + cmp esi, [esp] + ja .file_done +; 8c. For every assembled line, look at first token; +; go to 8d for token 'file', +; go to 8e for token 'format', +; go to 8f for token 'section', +; continue the loop otherwise. + mov eax, [esi+ebx+asm_row.preproc_offs] + add eax, [ebx+fas_header.preproc_offs] + cmp byte [eax+ebx+preproc_line_header.contents], 1Ah + jnz .file_next + movzx ecx, byte [eax+ebx+preproc_line_header.contents+1] + cmp cl, 4 + jnz .file_no_file + cmp dword [eax+ebx+preproc_line_header.contents+2], 'file' + jnz .file_no_file4 +; 8d. For lines starting with 'file' token, loop over tokens and get names. +; Note that there can be several names in one line. +; Parsing of tokens is similar to step 5 with the difference that +; preprocessor token stops processing: 'file' directives are processed +; in assembler stage. +; We save/restore esi since it is used for another thing in the internal loop; +; we push eax, which currently contains ebx-relative pointer to +; preproc_line_header, to be able to access it from resolve_name. + push esi eax + lea esi, [eax+preproc_line_header.contents+6] +.file_loop_int: + mov al, [esi+ebx] + inc esi + test al, al + jz .file_loop_int_done + cmp al, ';' + jz .file_loop_int_done + cmp al, 1Ah + jz .fileword + cmp al, '"' + jnz .file_loop_int + call resolve_name + add esi, [esi+ebx-4] + jmp .file_loop_int +.fileword: + movzx eax, byte [esi+ebx] + lea esi, [esi+eax+1] + jmp .file_loop_int +.file_loop_int_done: + pop eax esi + jmp .file_next +.file_no_file4: + cmp dword [eax+ebx+preproc_line_header.contents+2], 'data' + jnz .file_next + jmp .file_scan_from +.file_no_file: + cmp cl, 6 + jnz .file_no_format + cmp dword [eax+ebx+preproc_line_header.contents+2], 'form' + jnz .file_no_format + cmp word [eax+ebx+preproc_line_header.contents+6], 'at' + jnz .file_no_format +; 8e. For lines starting with 'format' token, loop over tokens and look for stub name. +; Note that there can be another quoted string, namely file extension; +; stub name always follows the token 'on'. + mov edx, TokenOn + call scan_after_word + jmp .file_next +.file_no_format: + cmp cl, 7 + jnz .file_no_section + cmp dword [eax+ebx+preproc_line_header.contents+2], 'sect' + jnz .file_no_section + mov edx, dword [eax+ebx+preproc_line_header.contents+6] + and edx, 0FFFFFFh + cmp edx, 'ion' + jnz .file_no_section +.file_scan_from: + mov edx, TokenFrom + call scan_after_word +.file_no_section: +.file_next: + add esi, asm_row.sizeof + jmp .file_loop +.file_done: + pop edx +; 9. Write result. +; 9a. Append two newlines to the end of buffer. + stdcall alloc_in_buf, 2 + mov word [edi+ebx], 10 * 101h + inc edi + inc edi +; 9b. If '-e' option was given, duplicate dependencies list as fake goals +; = copy all of them and append ":\n\n". + cmp [.flags], 0 + jz .nodup + lea ecx, [edi+1] + mov esi, [.names] + sub ecx, esi + stdcall alloc_in_buf, ecx + add esi, ebx + add edi, ebx + rep movsb + mov byte [edi-3], ':' + mov word [edi-2], 10 * 101h + sub edi, ebx +.nodup: +; 9c. Write to output file. + mov eax, [.outstart] + sub edi, eax + add eax, ebx + call write +; 10. Exit. + xor eax, eax + call exit + +; Helper procedure for steps 8e and 8f of main algorithm. +; Looks for quoted strings after given word in edx. +scan_after_word: + push esi eax + lea esi, [eax+preproc_line_header.contents+2+ecx] +.loop: + xor ecx, ecx +.loop_afterword: + mov al, [esi+ebx] + inc esi + test al, al + jz .loop_done + cmp al, ';' + jz .loop_done + cmp al, 1Ah + jz .word + cmp al, '"' + jnz .loop + test ecx, ecx + jz .skip_quoted + call resolve_name +.loop_done: + pop eax esi + ret +.skip_quoted: + add esi, [esi+ebx] + add esi, 4 + jmp .loop +.word: + movzx ecx, byte [esi+ebx] + lea esi, [esi+ecx+1] + cmp cl, byte [edx] + jnz .loop + push esi edi + add esi, ebx + sub esi, ecx + lea edi, [edx+1] + repz cmpsb + pop edi esi + jnz .loop + dec ecx + jmp .loop_afterword + +; Helper procedure for step 6 of the main procedure. +; Copies the ASCIIZ name from strings section to the buffer. +copy_asciiz_escaped: + add esi, [ebx+fas_header.strings_offs] +.loop: + mov al, [esi+ebx] + test al, al + jz .nothing + call copy_char_escaped + jmp .loop +.nothing: + ret + +; Helper procedure for step 7 of the main procedure. +; Copies the name with known length to the buffer. +copy_name_escaped: + mov ecx, [esi+ebx] + add esi, 4 + test ecx, ecx + jz .nothing + push ecx +.loop: + mov al, [esi+ebx] + call copy_char_escaped + dec dword [esp] + jnz .loop + pop ecx +.nothing: + ret + +; Helper procedure for steps 7 and 8 of the main procedure. +; Writes separator of file names in output = " \\\n". +add_separator: + stdcall alloc_in_buf, 3 + mov word [edi+ebx], ' \' + mov byte [edi+ebx+2], 10 + add edi, 3 + ret + +; Helper procedure for step 7 of the main procedure. +; Resolves the path to 'file' dependency and copies +; the full name to the buffer. +resolve_name: +; FASM uses the following order to search for referenced files: +; * path of currently assembling file, which may be .asm or .inc +; * paths from %INCLUDE% for versions >= 1.70 +; * current directory = file name is taken as is, without prepending dir name +; We mirror this behaviour, trying to find an existing file somewhere. +; There can be following reasons for the file can not be found anywhere: +; * it has been deleted between compilation and our actions +; * it didn't exist at all, compilation has failed +; * we are running in environment different from fasm environment. +; Assume that the last reason is most probable and that invalid dependency +; is better than absent dependency (it is easier to fix an explicit error +; than a silent one) and output file name without prepending dir name, +; as in the last case. Actually, we even don't need to test existence +; of the file in the current directory. + add esi, 4 ; skip string length +; 1. Get ebx-relative pointer to preproc_line_header, see the comment in start.7d + mov eax, [esp+4] +; 2. Get the path to currently processing file. + push esi +.getpath: + test byte [eax+ebx+preproc_line_header.line_number+3], 80h + jz @f + mov eax, [eax+ebx+preproc_line_header.line_offset] + add eax, [ebx+fas_header.preproc_offs] + jmp .getpath +@@: + mov edx, [eax+ebx+preproc_line_header.source_name] + test edx, edx + jz .frommain + add edx, [ebx+fas_header.preproc_offs] + jmp @f +.frommain: + mov edx, [ebx+fas_header.input] + add edx, [ebx+fas_header.strings_offs] +@@: +; 3. Check that it is not a duplicate of the previous dependency. +; 3a. Compare preprocessor units. + cmp edx, [start.prevfilefrom] + jnz .nodup +; 3b. Compare string lengths. + mov eax, [start.prevfile] + mov ecx, [eax+ebx-4] + cmp ecx, [esi+ebx-4] + jnz .nodup +; 3c. Compare string contents. + push esi edi + lea edi, [eax+ebx] + add esi, ebx + rep cmpsb + pop edi esi + jnz .nodup +; 3d. It is duplicate, just return. + pop esi + ret +.nodup: +; 3e. It is not duplicate. Output separator. + mov [start.prevfilefrom], edx + mov [start.prevfile], esi + call add_separator +; 4. Cut the last component of the path found in step 2. + mov ecx, edx + mov esi, edx +.scanpath: + mov al, [edx+ebx] + test al, al + jz .scandone + cmp al, '/' + jz .slash + cmp al, '\' + jnz .scannext +.slash: + lea ecx, [edx+1] +.scannext: + inc edx + jmp .scanpath +.scandone: + sub ecx, esi +; 5. Try path found in step 4. If found, go to step 8. + mov [start.testname], edi + stdcall copy_string, esi, ecx + pop esi + call expand_environment + call test_file_exists + test eax, eax + jns .found + call revert_testname +; 6. Try each of include paths. For every path, if file is found, go to step 8. +; Otherwise, continue loop over include path. +; Skip this step before 1.70. + cmp [ebx+fas_header.major], 1 + ja .checkenv + jb .nocheckenv + cmp [ebx+fas_header.minor], 70 + jb .nocheckenv +.checkenv: + mov ecx, [start.include] +.includeloop_ext: + mov eax, ecx +.includeloop_int: + cmp byte [eax+ebx], 0 + jz @f + cmp byte [eax+ebx], ';' + jz @f + inc eax + jmp .includeloop_int +@@: + push eax + sub eax, ecx + jz @f + stdcall copy_string, ecx, eax + cmp byte [edi+ebx-1], '/' + jz .hasslash +@@: + stdcall alloc_in_buf, 1 + mov byte [edi+ebx], '/' + inc edi +.hasslash: + call expand_environment + call test_file_exists + pop ecx + test eax, eax + jns .found + call revert_testname + cmp byte [ecx+ebx], 0 + jz .notfound + inc ecx + cmp byte [ecx+ebx], 0 + jnz .includeloop_ext +.nocheckenv: +.notfound: +; 7. File not found neither near the current preprocessor unit nor in %INCLUDE%. +; Assume that it is in the current directory. + call expand_environment +.found: +; 8. Currently we have file name from [start.testname] to edi; +; it is zero-terminated and not space-escaped. Fix both issues. + dec edi + inc [start.free] + push esi + mov edx, [start.testname] +.escapeloop: + cmp edx, edi + jae .escapedone + cmp byte [edx+ebx], ' ' + jnz .noescape + stdcall alloc_in_buf, 1 + mov ecx, edi + sub ecx, edx + push edi + add edi, ebx + lea esi, [edi-1] + std + rep movsb + cld + pop edi + inc edi + mov byte [edx+ebx], '\' + inc edx +.noescape: + inc edx + jmp .escapeloop +.escapedone: + pop esi + ret + +; Helper procedure for resolve_name. +; Allocates space in the buffer and appends the given string to the buffer. +copy_string: + mov eax, [esp+8] + test eax, eax + jz .nothing + stdcall alloc_in_buf, eax + mov ecx, [esp+4] +.copy: + mov al, [ecx+ebx] + inc ecx + cmp al, '\' + jnz @f + mov al, '/' +@@: + mov [edi+ebx], al + inc edi + dec dword [esp+8] + jnz .copy +.nothing: + ret 8 + +; Helper procedure for resolve_name. Undoes appending of last file name. +revert_testname: + add [start.free], edi + mov edi, [start.testname] + sub [start.free], edi + ret + +; Helper procedure for resolve_name. Copies string from esi to edi, +; expanding environment variables. +expand_environment: +; 1. Save esi to restore it in the end of function. + push esi +; 2. Push string length to the stack to be used as a variable. + pushd [esi+ebx-4] +; 3. Scan loop. +.scan: +; 3a. Scan for '%' sign. + call find_percent +.justcopy: +; 3b. Copy the part from the beginning of current portion to '%' sign, +; advance pointer to '%' sign, or end-of-string if no '%' found. + push eax + sub eax, esi + stdcall copy_string, esi, eax + pop esi +; 3c. If string has ended, break from the loop. + cmp dword [esp], 0 + jz .scandone +; 3d. Advance over '%' sign. + inc esi + dec dword [esp] +; 3e. Find paired '%'. + call find_percent +; 3f. If there is no paired '%', just return to 3b and copy remaining data, +; including skipped '%'; after that, 3c would break from the loop. + dec esi + cmp dword [esp], 0 + jz .justcopy +; 3g. Otherwise, get the value of environment variable. +; Since get_environment_variable requires zero-terminated string +; and returns zero-terminated string, temporarily overwrite trailing '%' +; and ignore last byte in returned string. +; Also convert any backslashes to forward slashes. + inc esi + mov byte [eax+ebx], 0 + push eax + push edi + call get_environment_variable + dec edi + inc [start.free] + pop eax +.replaceslash: + cmp eax, edi + jz .replaceslash_done + cmp byte [eax+ebx], '\' + jnz @f + mov byte [eax+ebx], '/' +@@: + inc eax + jmp .replaceslash +.replaceslash_done: + pop esi + mov byte [esi+ebx], '%' +; 3h. Advance over trailing '%'. + inc esi + dec dword [esp] +; 3i. Continue the loop. + jmp .scan +.scandone: +; 4. Zero-terminate resulting string. + stdcall alloc_in_buf, 1 + mov byte [edi+ebx], 0 + inc edi +; 5. Pop stack variable initialized in step 2. + pop eax +; 6. Restore esi saved in step 1 and return. + pop esi + ret + +; Helper procedure for expand_environment. +; Scans the string in esi with length [esp+4] +; until '%' is found or line ended. +find_percent: + mov eax, esi + cmp dword [esp+4], 0 + jz .nothing +.scan: + cmp byte [eax+ebx], '%' + jz .nothing + inc eax + dec dword [esp+4] + jnz .scan +.nothing: + ret + +; Helper procedure for copy_{name,asciiz}_escaped. +; Allocates space and writes one character, possibly escaped. +copy_char_escaped: + cmp al, ' ' + jnz .noescape + stdcall alloc_in_buf, 1 + mov byte [edi+ebx], '\' + inc edi +.noescape: + stdcall alloc_in_buf, 1 + mov al, [esi+ebx] + inc esi + cmp al, '\' + jnz @f + mov al, '/' +@@: + mov [edi+ebx], al + inc edi + ret + +; Helper procedure for ensuring that there is at least [esp+4] +; free bytes in the buffer. +alloc_in_buf: + mov eax, [esp+4] + sub [start.free], eax + jb .need_realloc + ret 4 +.need_realloc: + mov eax, [start.allocated] + add eax, eax + push ecx edx + call realloc + pop edx ecx + cmp [start.free], 0 + jl .need_realloc + mov ebx, [start.buf] + ret 4 + +badfile: + mov esi, badfile_string + call sayerr + mov al, 1 + call exit + +information: + mov esi, information_string + call sayerr + mov al, 2 + call exit + +nomemory: + mov esi, nomemory_string + call sayerr + mov al, 3 + call exit + +in_openerr: + mov esi, in_openerr_string + jmp in_err +readerr: + mov esi, readerr_string +in_err: + call sayerr + mov al, 4 + call exit + +out_openerr: + mov esi, out_openerr_string + jmp out_err +writeerr: + mov esi, writeerr_string +out_err: + call sayerr + mov al, 5 + call exit + +; Platform-specific procedures. +match =WINDOWS,OS { include 'windows_sys.inc' } +match =LINUX,OS { include 'linux_sys.inc' } + +; Data +macro string a, [b] { +common + db a ## _end - a +a db b +a ## _end: +} + +string information_string, 'Usage: fasmdep [-e] [ []]',10 +string badfile_string, 'Not .fas file',10 +string nomemory_string, 'No memory',10 +string in_openerr_string, 'Cannot open input file',10 +string readerr_string, 'Read error',10 +string out_openerr_string, 'Cannot create output file',10 +string writeerr_string, 'Write error',10 + +include_variable db 'INCLUDE',0 +TokenOn db 2,'on' +TokenFrom db 4,'from' diff --git a/data/common/fasmdep/linux_header.inc b/data/common/fasmdep/linux_header.inc new file mode 100644 index 000000000..cbbddf445 --- /dev/null +++ b/data/common/fasmdep/linux_header.inc @@ -0,0 +1,73 @@ +; Header for Linux program + format ELF executable 3 + entry start +; for system calls +include 'unistd.inc' +macro __mov a,b +{ +if b eq +else if ~(b eqtype 1) + mov a, b +else if b = 0 + xor a, a +else if (b < 0x80) & (b >= -0x80) + push b + pop a +else + mov a, b +end if +} +macro kercall a,b,c,d,e,f,g +{ + __mov eax, a + __mov ebx, b + __mov ecx, c + __mov edx, d + __mov esi, e + __mov edi, f + __mov ebp, g + int 0x80 +} +macro stdcall func,[arg] +{ +reverse + pushd arg +common + call func +} +PROT_READ = 0x1 ; page can be read +PROT_WRITE = 0x2 ; page can be written +PROT_EXEC = 0x4 ; page can be executed +PROT_SEM = 0x8 ; page may be used for atomic ops +PROT_NONE = 0x0 ; page can not be accessed +PROT_GROWSDOWN = 0x01000000 ; mprotect flag: extend change to start of growsdown vma +PROT_GROWSUP = 0x02000000 ; mprotect flag: extend change to end of growsup vma + +MAP_SHARED = 0x01 ; Share changes +MAP_PRIVATE = 0x02 ; Changes are private +MAP_TYPE = 0x0f ; Mask for type of mapping +MAP_FIXED = 0x10 ; Interpret addr exactly +MAP_ANONYMOUS = 0x20 ; don't use a file + +O_ACCMODE = 00000003 +O_RDONLY = 00000000 +O_WRONLY = 00000001 +O_RDWR = 00000002 +O_CREAT = 00000100 ; not fcntl +O_EXCL = 00000200 ; not fcntl +O_NOCTTY = 00000400 ; not fcntl +O_TRUNC = 00001000 ; not fcntl +O_APPEND = 00002000 +O_NONBLOCK = 00004000 +O_DSYNC = 00010000 ; used to be O_SYNC, see below +FASYNC = 00020000 ; fcntl, for BSD compatibility +O_DIRECT = 00040000 ; direct disk access hint +O_LARGEFILE = 00100000 +O_DIRECTORY = 00200000 ; must be a directory +O_NOFOLLOW = 00400000 ; don't follow links +O_NOATIME = 01000000 +O_CLOEXEC = 02000000 ; set close_on_exec +__O_SYNC = 04000000 +O_SYNC = (__O_SYNC + O_DSYNC) +O_NDELAY = O_NONBLOCK +segment readable executable diff --git a/data/common/fasmdep/linux_sys.inc b/data/common/fasmdep/linux_sys.inc new file mode 100644 index 000000000..48aa8d00b --- /dev/null +++ b/data/common/fasmdep/linux_sys.inc @@ -0,0 +1,186 @@ +; Platform-specific procedures for Linux. + +; Reallocate memory at pointer [start.buf] and size [start.allocated], +; new size is in eax. +realloc: + push ebx esi edi + mov ecx, eax + push ebp + kercall __NR_mmap2, 0, , PROT_READ+PROT_WRITE, MAP_PRIVATE+MAP_ANONYMOUS, -1, 0 + pop ebp + cmp eax, 0xFFFFF000 + ja nomemory + add [start.free], ecx + xchg ecx, [start.allocated] + sub [start.free], ecx + mov edi, eax + xchg eax, [start.buf] + shr ecx, 2 + jz .nothing + push ecx + mov esi, eax + rep movsd + pop ecx + shl ecx, 2 + call free_eax_ecx +.nothing: + pop edi esi ebx + ret + +; Read the next portion of input data to [start.buf]. +read: + mov ecx, [start.buf] + add ecx, [start.allocated] + mov edx, [start.free] + sub ecx, edx + kercall __NR_read, [start.in], , + test eax, eax + js readerr + ret + +; Write output data: eax=pointer, edi=size. +write: + mov ecx, eax + kercall __NR_write, [start.out], , edi + cmp eax, edi + jnz writeerr + ret + +; Parse command line, open input and output files. +get_params: +; 1. Initialize default streams: 0 for stdin, 1 for stdout. + inc [start.out] +; 2. Prepare for scanning, skipping argv[0]. + mov ebx, [start.argc] + cmp ebx, 1 + jbe .noargs + dec ebx + lea esi, [start.argv+4] ; skip argv[0] + xor edi, edi ; no args parsed yet +; 3. Parse loop. +.parse: +; 3a. Get the next argument. + lodsd +; 3b. Check whether it is a known option. + cmp word [eax], '-e' + jnz @f + cmp byte [eax+2], 0 + jnz @f +; 3c. If it is, modify flags and continue the loop. + mov [start.flags], 1 ; '-e' is given + jmp .nextarg +@@: +; 3d. Otherwise, it is a name of input or output file. +; edi keeps the count of names encountered before; +; edi = 0 means this is input file, edi = 1 - output file, +; otherwise this is third arg, which is an error + cmp edi, 1 + ja information +; 3e. Some parameters of __NR_open differ for input and output. Setup them. + mov ecx, O_WRONLY+O_CREAT+O_TRUNC + mov edx, 0644o + jz @f + mov ecx, O_RDONLY +@@: +; 3f. Open/create the file, save the handle. + push ebx + mov ebx, eax + kercall __NR_open + pop ebx + test eax, eax + js .fileerr + mov [start.in+edi*4], eax + inc edi +.nextarg: + dec ebx + jnz .parse +.noargs: +; 4. End of command line reached. Return. + ret +.fileerr: + test edi, edi + jz in_openerr + jmp out_openerr + +; Exit, return code is in al. +exit: + movzx ebx, al + push ebx + mov eax, [start.buf] + test eax, eax + jz @f + mov ecx, [start.allocated] + call free_eax_ecx +@@: + kercall __NR_close, [start.in] + kercall __NR_close, [start.out] + pop ebx + kercall __NR_exit + +; Helper procedure for realloc and exit. +free_eax_ecx: + mov ebx, eax + kercall __NR_munmap + ret + +; Output the message given in esi to stderr. +sayerr: + movzx edx, byte [esi-1] + kercall __NR_write, 2, esi, + ret + +; Get environment variable esi (ebx-relative pointer) to the buffer, +; expanding it if needed. +get_environment_variable: + mov ecx, [start.argc] + lea ecx, [start.argv+ecx*4+4] +.findvar: + mov edx, [ecx] + add ecx, 4 + test edx, edx + jz .notfound + push esi + add esi, ebx +.comparename: + lodsb + cmp al, [edx] + jnz @f + inc edx + jmp .comparename +@@: + pop esi + test al, al + jnz .findvar + cmp byte [edx], '=' + jnz .findvar + inc edx + xor eax, eax +@@: + inc eax + cmp byte [edx+eax-1], 0 + jnz @b + stdcall alloc_in_buf, eax +@@: + mov al, [edx] + inc edx + mov [edi+ebx], al + inc edi + test al, al + jnz @b + ret +.notfound: + stdcall alloc_in_buf, 1 + mov byte [edi+ebx], 0 + inc edi + ret + +; Test whether a file with name [.testname] exists. +; Returns eax<0 if not, nonzero otherwise. +test_file_exists: + push ebx + add ebx, [start.testname] + sub esp, 800h + kercall __NR_stat, , esp + add esp, 800h + pop ebx + ret diff --git a/data/common/fasmdep/unistd.inc b/data/common/fasmdep/unistd.inc new file mode 100644 index 000000000..cefad1083 --- /dev/null +++ b/data/common/fasmdep/unistd.inc @@ -0,0 +1,322 @@ +__NR_restart_syscall = 0 +__NR_exit = 1 +__NR_fork = 2 +__NR_read = 3 +__NR_write = 4 +__NR_open = 5 +__NR_close = 6 +__NR_waitpid = 7 +__NR_creat = 8 +__NR_link = 9 +__NR_unlink = 10 +__NR_execve = 11 +__NR_chdir = 12 +__NR_time = 13 +__NR_mknod = 14 +__NR_chmod = 15 +__NR_lchown = 16 +__NR_break = 17 +__NR_oldstat = 18 +__NR_lseek = 19 +__NR_getpid = 20 +__NR_mount = 21 +__NR_umount = 22 +__NR_setuid = 23 +__NR_getuid = 24 +__NR_stime = 25 +__NR_ptrace = 26 +__NR_alarm = 27 +__NR_oldfstat = 28 +__NR_pause = 29 +__NR_utime = 30 +__NR_stty = 31 +__NR_gtty = 32 +__NR_access = 33 +__NR_nice = 34 +__NR_ftime = 35 +__NR_sync = 36 +__NR_kill = 37 +__NR_rename = 38 +__NR_mkdir = 39 +__NR_rmdir = 40 +__NR_dup = 41 +__NR_pipe = 42 +__NR_times = 43 +__NR_prof = 44 +__NR_brk = 45 +__NR_setgid = 46 +__NR_getgid = 47 +__NR_signal = 48 +__NR_geteuid = 49 +__NR_getegid = 50 +__NR_acct = 51 +__NR_umount2 = 52 +__NR_lock = 53 +__NR_ioctl = 54 +__NR_fcntl = 55 +__NR_mpx = 56 +__NR_setpgid = 57 +__NR_ulimit = 58 +__NR_oldolduname = 59 +__NR_umask = 60 +__NR_chroot = 61 +__NR_ustat = 62 +__NR_dup2 = 63 +__NR_getppid = 64 +__NR_getpgrp = 65 +__NR_setsid = 66 +__NR_sigaction = 67 +__NR_sgetmask = 68 +__NR_ssetmask = 69 +__NR_setreuid = 70 +__NR_setregid = 71 +__NR_sigsuspend = 72 +__NR_sigpending = 73 +__NR_sethostname = 74 +__NR_setrlimit = 75 +__NR_getrlimit = 76 +__NR_getrusage = 77 +__NR_gettimeofday = 78 +__NR_settimeofday = 79 +__NR_getgroups = 80 +__NR_setgroups = 81 +__NR_select = 82 +__NR_symlink = 83 +__NR_oldlstat = 84 +__NR_readlink = 85 +__NR_uselib = 86 +__NR_swapon = 87 +__NR_reboot = 88 +__NR_readdir = 89 +__NR_mmap = 90 +__NR_munmap = 91 +__NR_truncate = 92 +__NR_ftruncate = 93 +__NR_fchmod = 94 +__NR_fchown = 95 +__NR_getpriority = 96 +__NR_setpriority = 97 +__NR_profil = 98 +__NR_statfs = 99 +__NR_fstatfs = 100 +__NR_ioperm = 101 +__NR_socketcall = 102 +__NR_syslog = 103 +__NR_setitimer = 104 +__NR_getitimer = 105 +__NR_stat = 106 +__NR_lstat = 107 +__NR_fstat = 108 +__NR_olduname = 109 +__NR_iopl = 110 +__NR_vhangup = 111 +__NR_idle = 112 +__NR_vm86old = 113 +__NR_wait4 = 114 +__NR_swapoff = 115 +__NR_sysinfo = 116 +__NR_ipc = 117 +__NR_fsync = 118 +__NR_sigreturn = 119 +__NR_clone = 120 +__NR_setdomainname = 121 +__NR_uname = 122 +__NR_modify_ldt = 123 +__NR_adjtimex = 124 +__NR_mprotect = 125 +__NR_sigprocmask = 126 +__NR_create_module = 127 +__NR_init_module = 128 +__NR_delete_module = 129 +__NR_get_kernel_syms = 130 +__NR_quotactl = 131 +__NR_getpgid = 132 +__NR_fchdir = 133 +__NR_bdflush = 134 +__NR_sysfs = 135 +__NR_personality = 136 +__NR_afs_syscall = 137 +__NR_setfsuid = 138 +__NR_setfsgid = 139 +__NR__llseek = 140 +__NR_getdents = 141 +__NR__newselect = 142 +__NR_flock = 143 +__NR_msync = 144 +__NR_readv = 145 +__NR_writev = 146 +__NR_getsid = 147 +__NR_fdatasync = 148 +__NR__sysctl = 149 +__NR_mlock = 150 +__NR_munlock = 151 +__NR_mlockall = 152 +__NR_munlockall = 153 +__NR_sched_setparam = 154 +__NR_sched_getparam = 155 +__NR_sched_setscheduler = 156 +__NR_sched_getscheduler = 157 +__NR_sched_yield = 158 +__NR_sched_get_priority_max = 159 +__NR_sched_get_priority_min = 160 +__NR_sched_rr_get_interval = 161 +__NR_nanosleep = 162 +__NR_mremap = 163 +__NR_setresuid = 164 +__NR_getresuid = 165 +__NR_vm86 = 166 +__NR_query_module = 167 +__NR_poll = 168 +__NR_nfsservctl = 169 +__NR_setresgid = 170 +__NR_getresgid = 171 +__NR_prctl = 172 +__NR_rt_sigreturn = 173 +__NR_rt_sigaction = 174 +__NR_rt_sigprocmask = 175 +__NR_rt_sigpending = 176 +__NR_rt_sigtimedwait = 177 +__NR_rt_sigqueueinfo = 178 +__NR_rt_sigsuspend = 179 +__NR_pread64 = 180 +__NR_pwrite64 = 181 +__NR_chown = 182 +__NR_getcwd = 183 +__NR_capget = 184 +__NR_capset = 185 +__NR_sigaltstack = 186 +__NR_sendfile = 187 +__NR_getpmsg = 188 +__NR_putpmsg = 189 +__NR_vfork = 190 +__NR_ugetrlimit = 191 +__NR_mmap2 = 192 +__NR_truncate64 = 193 +__NR_ftruncate64 = 194 +__NR_stat64 = 195 +__NR_lstat64 = 196 +__NR_fstat64 = 197 +__NR_lchown32 = 198 +__NR_getuid32 = 199 +__NR_getgid32 = 200 +__NR_geteuid32 = 201 +__NR_getegid32 = 202 +__NR_setreuid32 = 203 +__NR_setregid32 = 204 +__NR_getgroups32 = 205 +__NR_setgroups32 = 206 +__NR_fchown32 = 207 +__NR_setresuid32 = 208 +__NR_getresuid32 = 209 +__NR_setresgid32 = 210 +__NR_getresgid32 = 211 +__NR_chown32 = 212 +__NR_setuid32 = 213 +__NR_setgid32 = 214 +__NR_setfsuid32 = 215 +__NR_setfsgid32 = 216 +__NR_pivot_root = 217 +__NR_mincore = 218 +__NR_madvise = 219 +__NR_madvise1 = 219 +__NR_getdents64 = 220 +__NR_fcntl64 = 221 +__NR_gettid = 224 +__NR_readahead = 225 +__NR_setxattr = 226 +__NR_lsetxattr = 227 +__NR_fsetxattr = 228 +__NR_getxattr = 229 +__NR_lgetxattr = 230 +__NR_fgetxattr = 231 +__NR_listxattr = 232 +__NR_llistxattr = 233 +__NR_flistxattr = 234 +__NR_removexattr = 235 +__NR_lremovexattr = 236 +__NR_fremovexattr = 237 +__NR_tkill = 238 +__NR_sendfile64 = 239 +__NR_futex = 240 +__NR_sched_setaffinity = 241 +__NR_sched_getaffinity = 242 +__NR_set_thread_area = 243 +__NR_get_thread_area = 244 +__NR_io_setup = 245 +__NR_io_destroy = 246 +__NR_io_getevents = 247 +__NR_io_submit = 248 +__NR_io_cancel = 249 +__NR_fadvise64 = 250 +__NR_exit_group = 252 +__NR_lookup_dcookie = 253 +__NR_epoll_create = 254 +__NR_epoll_ctl = 255 +__NR_epoll_wait = 256 +__NR_remap_file_pages = 257 +__NR_set_tid_address = 258 +__NR_timer_create = 259 +__NR_statfs64 = 268 +__NR_fstatfs64 = 269 +__NR_tgkill = 270 +__NR_utimes = 271 +__NR_fadvise64_64 = 272 +__NR_vserver = 273 +__NR_mbind = 274 +__NR_get_mempolicy = 275 +__NR_set_mempolicy = 276 +__NR_mq_open = 277 +__NR_kexec_load = 283 +__NR_waitid = 284 +__NR_sys_setaltroot = 285 +__NR_add_key = 286 +__NR_request_key = 287 +__NR_keyctl = 288 +__NR_ioprio_set = 289 +__NR_ioprio_get = 290 +__NR_inotify_init = 291 +__NR_inotify_add_watch = 292 +__NR_inotify_rm_watch = 293 +__NR_migrate_pages = 294 +__NR_openat = 295 +__NR_mkdirat = 296 +__NR_mknodat = 297 +__NR_fchownat = 298 +__NR_futimesat = 299 +__NR_fstatat64 = 300 +__NR_unlinkat = 301 +__NR_renameat = 302 +__NR_linkat = 303 +__NR_symlinkat = 304 +__NR_readlinkat = 305 +__NR_fchmodat = 306 +__NR_faccessat = 307 +__NR_pselect6 = 308 +__NR_ppoll = 309 +__NR_unshare = 310 +__NR_set_robust_list = 311 +__NR_get_robust_list = 312 +__NR_splice = 313 +__NR_sync_file_range = 314 +__NR_tee = 315 +__NR_vmsplice = 316 +__NR_move_pages = 317 +__NR_getcpu = 318 +__NR_epoll_pwait = 319 +__NR_utimensat = 320 +__NR_signalfd = 321 +__NR_timerfd_create = 322 +__NR_eventfd = 323 +__NR_fallocate = 324 +__NR_timerfd_settime = 325 +__NR_timerfd_gettime = 326 +__NR_signalfd4 = 327 +__NR_eventfd2 = 328 +__NR_epoll_create1 = 329 +__NR_dup3 = 330 +__NR_pipe2 = 331 +__NR_inotify_init1 = 332 +__NR_preadv = 333 +__NR_pwritev = 334 +__NR_rt_tgsigqueueinfo = 335 +__NR_perf_event_open = 336 diff --git a/data/common/fasmdep/windows_header.inc b/data/common/fasmdep/windows_header.inc new file mode 100644 index 000000000..41493b6a0 --- /dev/null +++ b/data/common/fasmdep/windows_header.inc @@ -0,0 +1,6 @@ +; Header for Windows program + format PE console +include 'win32w.inc' +ERROR_BROKEN_PIPE = 109 +ERROR_MORE_DATA = 234 +section '.text' code readable executable diff --git a/data/common/fasmdep/windows_sys.inc b/data/common/fasmdep/windows_sys.inc new file mode 100644 index 000000000..444259a92 --- /dev/null +++ b/data/common/fasmdep/windows_sys.inc @@ -0,0 +1,225 @@ +; Platform-specific procedures for Windows. + +; Reallocate memory at pointer [start.buf] and size [start.allocated], +; new size is in eax. +realloc: + push eax + stdcall [VirtualAlloc], 0, eax, MEM_COMMIT + MEM_RESERVE, PAGE_READWRITE + pop ecx + test eax, eax + jz nomemory + add [start.free], ecx + xchg ecx, [start.allocated] + sub [start.free], ecx + push esi edi + mov edi, eax + xchg eax, [start.buf] + shr ecx, 2 + jz .nothing + mov esi, eax + rep movsd + call free_eax +.nothing: + pop edi esi + ret + +; Read the next portion of input data to [start.buf]. +read: + push eax ; reserve space for *lpNumberOfBytesRead + mov ecx, esp + mov eax, [start.buf] + add eax, [start.allocated] + sub eax, [start.free] + stdcall [ReadFile], [start.in], eax, [start.free], ecx, 0 + test eax, eax + jz @f +.nothing: + pop eax + ret +@@: +; ERROR_BROKEN_PIPE and ERROR_MORE_DATA are normal codes when dealing with pipes + call [GetLastError] + cmp eax, ERROR_BROKEN_PIPE + jz .nothing + cmp eax, ERROR_MORE_DATA + jz .nothing + pop eax + jmp readerr + +; Write output data: eax=pointer, edi=size. +write: + push eax + mov ecx, esp + stdcall [WriteFile], [start.out], eax, edi, ecx, 0 + test eax, eax + pop eax + jz writeerr + cmp eax, edi + jnz writeerr + ret + +; Parse command line, open input and output files. +get_params: +; 1. Get the command line split to argv[] array. + call [GetCommandLineW] + push eax ; reserve space for *pNumArgs + stdcall [CommandLineToArgvW], eax, esp + pop ebx ; ebx = argc, eax = argv + push eax ; save argument for [LocalFree] +; 2. Prepare for scanning, skipping argv[0]. + cmp ebx, 1 + jbe .noargs + dec ebx + lea esi, [eax+4] ; skip argv[0] + xor edi, edi ; no args parsed yet +; 3. Parse loop. +.parse: +; 3a. Get the next argument. + lodsd +; 3b. Check whether it is a known option. + cmp dword [eax], '-' + 'e' * 65536 + jnz @f + cmp word [eax+4], 0 + jnz @f +; 3c. If it is, modify flags and continue the loop. + mov [start.flags], 1 ; '-e' is given + jmp .nextarg +@@: +; 3d. Otherwise, it is a name of input or output file. +; edi keeps the count of names encountered before; +; edi = 0 means this is input file, edi = 1 - output file, +; otherwise this is third arg, which is an error + cmp edi, 1 + ja .toomanyargs +; 3e. Some parameters of CreateFileW differ for input and output. Setup them. + mov ecx, GENERIC_WRITE + mov edx, CREATE_ALWAYS + jz @f + add ecx, ecx ; GENERIC_READ + inc edx ; OPEN_EXISTING +@@: +; 3f. Open/create the file, save the handle. + stdcall [CreateFileW], eax, ecx, FILE_SHARE_READ+FILE_SHARE_DELETE, 0, edx, FILE_ATTRIBUTE_NORMAL, 0 + cmp eax, INVALID_HANDLE_VALUE + jz .fileerr + mov [start.in+edi*4], eax + inc edi +.nextarg: + dec ebx + jnz .parse +.noargs: +; 4. End of command line reached. If input and/or output was not given, use defaults. + test edi, edi + jnz .hasinput + stdcall [GetStdHandle], STD_INPUT_HANDLE + mov [start.in], eax +.hasinput: + cmp edi, 1 + ja .hasoutput + stdcall [GetStdHandle], STD_OUTPUT_HANDLE + mov [start.out], eax +.hasoutput: +; 5. Free memory allocated in step 1 and return. + call [LocalFree] + ret +.toomanyargs: + call [LocalFree] + jmp information +.fileerr: + call [LocalFree] + test edi, edi + jz in_openerr + jmp out_openerr + +; Exit, return code is in al. +exit: + movzx eax, al + push eax ; save return code for [ExitProcess] + mov eax, [start.buf] + test eax, eax + jz @f + call free_eax +@@: + stdcall [CloseHandle], [start.in] + stdcall [CloseHandle], [start.out] + call [ExitProcess] + +; Output the message given in esi to stderr. +sayerr: + stdcall [GetStdHandle], STD_ERROR_HANDLE + push eax + mov ecx, esp + movzx edx, byte [esi-1] + stdcall [WriteFile], eax, esi, edx, ecx, 0 + pop ecx + ret + +; Helper procedure for realloc and exit. +free_eax: + stdcall [VirtualFree], eax, 0, MEM_RELEASE + ret + +; Get environment variable esi (ebx-relative pointer) to the buffer, +; expanding it if needed. +get_environment_variable: + lea eax, [edi+ebx] + lea ecx, [esi+ebx] + stdcall [GetEnvironmentVariableA], ecx, eax, [start.free] +; GetEnvironmentVariable returns one of following values: +; * if all is ok: number of characters copied to the buffer, +; not including terminating zero +; * if buffer size is too small: number of characters required in the buffer, +; including terminating zero +; * if environment variable not found: zero +; We treat the last case the same as first one. + cmp eax, [start.free] + jae .resize + inc eax ; include terminating zero + add edi, eax + sub [start.free], eax + mov byte [edi+ebx-1], 0 ; force zero-terminated or empty string + ret +.resize: +; esi can be inside the buffer or outside the buffer; +; we must not correct it in the first case, +; but advance relative to new value of ebx in the second one. + mov ecx, esi + cmp esi, [start.allocated] + jb @f + add esi, ebx +@@: + stdcall alloc_in_buf, eax + cmp esi, ecx + jz get_environment_variable + sub esi, ebx + jmp get_environment_variable + +; Test whether a file with name [.testname] exists. +; Returns eax<0 if not, nonzero otherwise. +test_file_exists: + mov eax, [start.testname] + add eax, ebx + stdcall [GetFileAttributesA], eax + inc eax + ret + +; Imports +align 4 +data import +library kernel32,'kernel32.dll',shell32,'shell32.dll' +import kernel32, \ + GetLastError, 'GetLastError', \ + ExitProcess, 'ExitProcess', \ + VirtualAlloc, 'VirtualAlloc', \ + VirtualFree, 'VirtualFree', \ + LocalFree, 'LocalFree', \ + GetStdHandle, 'GetStdHandle', \ + CreateFileW, 'CreateFileW', \ + GetFileAttributesA, 'GetFileAttributesA', \ + ReadFile, 'ReadFile', \ + WriteFile, 'WriteFile', \ + CloseHandle, 'CloseHandle', \ + GetCommandLineW, 'GetCommandLineW', \ + GetEnvironmentVariableA, 'GetEnvironmentVariableA' +import shell32, CommandLineToArgvW, 'CommandLineToArgvW' +end data diff --git a/data/eng/Makefile b/data/eng/Makefile index a061377b3..772df2e9d 100644 --- a/data/eng/Makefile +++ b/data/eng/Makefile @@ -455,7 +455,7 @@ include Makefile.nasm include Makefile.copy # Special rules for copying sysfuncs.txt - it isn't directly included in the image. -docpack: $(DOCDIR)SYSFUNCS.TXT $(wildcard $(DOCDIR)*) +docpack: $(DOCDIR)SYSFUNCS.TXT $(DOCDIR)SYSFUNCS.TXT: $(KERNEL)/docs/sysfuncs.txt cp $(KERNEL)/docs/sysfuncs.txt $(DOCDIR)SYSFUNCS.TXT diff --git a/data/eng/Makefile.fasm b/data/eng/Makefile.fasm index 7e81ac593..67f4b4deb 100644 --- a/data/eng/Makefile.fasm +++ b/data/eng/Makefile.fasm @@ -28,22 +28,20 @@ # $(4) = name of program - without path and extension, define fasm_meta_rule $(1): $(2) Makefile.fasm .deps/.dir $$(call respace,$$(addsuffix .dir,$(3))) - fasm -m 65536 "$$<" "$$@" -s .deps/$(4).fas - prepsrc .deps/$(4).fas /dev/null - prepsrc .deps/$(4).fas /dev/stdout | \ - perl -n -e 's|\\|/|g;s| |\\ |g;push @a,$$$$1 if/^;include\\ \x27(.*?)\x27/;' \ - -e 'END{$$$$a=join " \\\n ",@a;print q`$(1): `,"$$$$a\n$$$$a:\n"}' > .deps/$(4).Po + tmpfile=`mktemp --tmpdir build.XXXXXXXX` && \ + (fasm -m 65536 "$$<" "$$@" -s $$$$tmpfile && \ + fasmdep -e $$$$tmpfile > .deps/$(4).Po && \ + rm $$$$tmpfile) || (rm $$$$tmpfile; false) kpack --nologo "$$@" -include .deps/$(4).Po endef define fasm_nokpack_meta_rule $(1): $(2) Makefile.fasm .deps/.dir $$(call respace,$$(addsuffix .dir,$(3))) - fasm -m 65536 "$$<" "$$@" -s .deps/$(4).fas - prepsrc .deps/$(4).fas /dev/null - prepsrc .deps/$(4).fas /dev/stdout | \ - perl -n -e 's|\\|/|g;s| |\\ |g;push @a,$$$$1 if/^;include\\ \x27(.*?)\x27/;' \ - -e 'END{$$$$a=join " \\\n ",@a;print q`$(1): `,"$$$$a\n$$$$a:\n"}' > .deps/$(4).Po + tmpfile=`mktemp --tmpdir build.XXXXXXXX` && \ + (fasm -m 65536 "$$<" "$$@" -s $$$$tmpfile && \ + fasmdep -e $$$$tmpfile > .deps/$(4).Po && \ + rm $$$$tmpfile) || (rm $$$$tmpfile; false) -include .deps/$(4).Po endef @@ -54,10 +52,8 @@ $(foreach f,$(FASM_NOKPACK_PROGRAMS),$(eval $(call fasm_nokpack_meta_rule,$(fbin # Rule for the kernel differs: it uses kerpack instead of kpack. kernel.mnt: $(KERNEL)/kernel.asm Makefile.fasm .deps/.dir - fasm -m 65536 "$<" "$@" -s .deps/kernel.fas - prepsrc .deps/kernel.fas /dev/null - prepsrc .deps/kernel.fas /dev/stdout | \ - perl -n -e 's|\\|/|g;s| |\\ |g;push @a,$$1 if/^;include\\ \x27(.*?)\x27/;' \ - -e 'END{$$a=join " \\\n ",@a;print "$@: $$a\n$$a:\n"}' > .deps/kernel.Po - kerpack $@ + tmpfile=`mktemp --tmpdir build.XXXXXXXX` && \ + (fasm -m 65536 "$<" "$@" -s $$tmpfile && \ + fasmdep -e $$tmpfile > .deps/kernel.Po && \ + rm $$tmpfile) || (rm $$tmpfile; false) -include .deps/kernel.Po diff --git a/data/new-stack/Makefile b/data/new-stack/Makefile index 1ae9f71b3..d06ba98b1 100644 --- a/data/new-stack/Makefile +++ b/data/new-stack/Makefile @@ -299,7 +299,7 @@ include Makefile.nasm include Makefile.copy # Special rules for copying sysfuncs.txt - it isn't directly included in the image. -docpack: $(DOCDIR)SYSFUNCS.TXT $(wildcard $(DOCDIR)*) +docpack: $(DOCDIR)SYSFUNCS.TXT $(DOCDIR)SYSFUNCS.TXT: $(TRUNKKERNEL)/docs/sysfuncs.txt cp $(TRUNKKERNEL)/docs/sysfuncs.txt $(DOCDIR)SYSFUNCS.TXT diff --git a/data/new-stack/Makefile.fasm b/data/new-stack/Makefile.fasm index 26e456d6e..67f4b4deb 100644 --- a/data/new-stack/Makefile.fasm +++ b/data/new-stack/Makefile.fasm @@ -28,25 +28,32 @@ # $(4) = name of program - without path and extension, define fasm_meta_rule $(1): $(2) Makefile.fasm .deps/.dir $$(call respace,$$(addsuffix .dir,$(3))) - fasm -m 65536 $$< "$$@" -s .deps/$(4).fas - prepsrc .deps/$(4).fas /dev/null - prepsrc .deps/$(4).fas /dev/stdout | \ - perl -n -e 's|\\|/|g;s| |\\ |g;push @a,$$$$1 if/^;include\\ \x27(.*?)\x27/;' \ - -e 'END{$$$$a=join " \\\n ",@a;print q`$(1): `,"$$$$a\n$$$$a:\n"}' > .deps/$(4).Po + tmpfile=`mktemp --tmpdir build.XXXXXXXX` && \ + (fasm -m 65536 "$$<" "$$@" -s $$$$tmpfile && \ + fasmdep -e $$$$tmpfile > .deps/$(4).Po && \ + rm $$$$tmpfile) || (rm $$$$tmpfile; false) kpack --nologo "$$@" -include .deps/$(4).Po endef +define fasm_nokpack_meta_rule +$(1): $(2) Makefile.fasm .deps/.dir $$(call respace,$$(addsuffix .dir,$(3))) + tmpfile=`mktemp --tmpdir build.XXXXXXXX` && \ + (fasm -m 65536 "$$<" "$$@" -s $$$$tmpfile && \ + fasmdep -e $$$$tmpfile > .deps/$(4).Po && \ + rm $$$$tmpfile) || (rm $$$$tmpfile; false) +-include .deps/$(4).Po +endef + progname=$(call respace,$(basename $(notdir $(call binarypart,$(f))))) binarydir=$(subst ./,,$(dir $(call binarypart,$(f)))) -$(foreach f,$(FASM_PROGRAMS) $(FASM_PROGRAMS_CD),$(eval $(call fasm_meta_rule,$(fbinary),$(fsource),$(binarydir),$(progname)))) +$(foreach f,$(FASM_PROGRAMS) $(FASM_PROGRAMS_CD) $(SKIN_SOURCES),$(eval $(call fasm_meta_rule,$(fbinary),$(fsource),$(binarydir),$(progname)))) +$(foreach f,$(FASM_NOKPACK_PROGRAMS),$(eval $(call fasm_nokpack_meta_rule,$(fbinary),$(fsource),$(binarydir),$(progname)))) # Rule for the kernel differs: it uses kerpack instead of kpack. kernel.mnt: $(KERNEL)/kernel.asm Makefile.fasm .deps/.dir - fasm -m 65536 $< "$@" -s .deps/kernel.fas - prepsrc .deps/kernel.fas /dev/null - prepsrc .deps/kernel.fas /dev/stdout | \ - perl -n -e 's|\\|/|g;s| |\\ |g;push @a,$$1 if/^;include\\ \x27(.*?)\x27/;' \ - -e 'END{$$a=join " \\\n ",@a;print "$@: $$a\n$$a:\n"}' > .deps/kernel.Po - kerpack $@ + tmpfile=`mktemp --tmpdir build.XXXXXXXX` && \ + (fasm -m 65536 "$<" "$@" -s $$tmpfile && \ + fasmdep -e $$tmpfile > .deps/kernel.Po && \ + rm $$tmpfile) || (rm $$tmpfile; false) -include .deps/kernel.Po diff --git a/data/rus/Makefile b/data/rus/Makefile index d5e69e483..fd997947e 100644 --- a/data/rus/Makefile +++ b/data/rus/Makefile @@ -474,7 +474,7 @@ include Makefile.nasm include Makefile.copy # Special rules for copying sysfuncr.txt - it isn't directly included in the image. -docpack: $(DOCDIR)SYSFUNCR.TXT $(wildcard $(DOCDIR)*) +docpack: $(DOCDIR)SYSFUNCR.TXT $(DOCDIR)SYSFUNCR.TXT: $(KERNEL)/docs/sysfuncr.txt cp $(KERNEL)/docs/sysfuncr.txt $(DOCDIR)SYSFUNCR.TXT diff --git a/data/rus/Makefile.fasm b/data/rus/Makefile.fasm index 7e81ac593..67f4b4deb 100644 --- a/data/rus/Makefile.fasm +++ b/data/rus/Makefile.fasm @@ -28,22 +28,20 @@ # $(4) = name of program - without path and extension, define fasm_meta_rule $(1): $(2) Makefile.fasm .deps/.dir $$(call respace,$$(addsuffix .dir,$(3))) - fasm -m 65536 "$$<" "$$@" -s .deps/$(4).fas - prepsrc .deps/$(4).fas /dev/null - prepsrc .deps/$(4).fas /dev/stdout | \ - perl -n -e 's|\\|/|g;s| |\\ |g;push @a,$$$$1 if/^;include\\ \x27(.*?)\x27/;' \ - -e 'END{$$$$a=join " \\\n ",@a;print q`$(1): `,"$$$$a\n$$$$a:\n"}' > .deps/$(4).Po + tmpfile=`mktemp --tmpdir build.XXXXXXXX` && \ + (fasm -m 65536 "$$<" "$$@" -s $$$$tmpfile && \ + fasmdep -e $$$$tmpfile > .deps/$(4).Po && \ + rm $$$$tmpfile) || (rm $$$$tmpfile; false) kpack --nologo "$$@" -include .deps/$(4).Po endef define fasm_nokpack_meta_rule $(1): $(2) Makefile.fasm .deps/.dir $$(call respace,$$(addsuffix .dir,$(3))) - fasm -m 65536 "$$<" "$$@" -s .deps/$(4).fas - prepsrc .deps/$(4).fas /dev/null - prepsrc .deps/$(4).fas /dev/stdout | \ - perl -n -e 's|\\|/|g;s| |\\ |g;push @a,$$$$1 if/^;include\\ \x27(.*?)\x27/;' \ - -e 'END{$$$$a=join " \\\n ",@a;print q`$(1): `,"$$$$a\n$$$$a:\n"}' > .deps/$(4).Po + tmpfile=`mktemp --tmpdir build.XXXXXXXX` && \ + (fasm -m 65536 "$$<" "$$@" -s $$$$tmpfile && \ + fasmdep -e $$$$tmpfile > .deps/$(4).Po && \ + rm $$$$tmpfile) || (rm $$$$tmpfile; false) -include .deps/$(4).Po endef @@ -54,10 +52,8 @@ $(foreach f,$(FASM_NOKPACK_PROGRAMS),$(eval $(call fasm_nokpack_meta_rule,$(fbin # Rule for the kernel differs: it uses kerpack instead of kpack. kernel.mnt: $(KERNEL)/kernel.asm Makefile.fasm .deps/.dir - fasm -m 65536 "$<" "$@" -s .deps/kernel.fas - prepsrc .deps/kernel.fas /dev/null - prepsrc .deps/kernel.fas /dev/stdout | \ - perl -n -e 's|\\|/|g;s| |\\ |g;push @a,$$1 if/^;include\\ \x27(.*?)\x27/;' \ - -e 'END{$$a=join " \\\n ",@a;print "$@: $$a\n$$a:\n"}' > .deps/kernel.Po - kerpack $@ + tmpfile=`mktemp --tmpdir build.XXXXXXXX` && \ + (fasm -m 65536 "$<" "$@" -s $$tmpfile && \ + fasmdep -e $$tmpfile > .deps/kernel.Po && \ + rm $$tmpfile) || (rm $$tmpfile; false) -include .deps/kernel.Po