xref: /freebsd/stand/forth/support.4th (revision c203bd70b5957f85616424b6fa374479372d06e3)
1\ Copyright (c) 1999 Daniel C. Sobral <dcs@FreeBSD.org>
2\ All rights reserved.
3\
4\ Redistribution and use in source and binary forms, with or without
5\ modification, are permitted provided that the following conditions
6\ are met:
7\ 1. Redistributions of source code must retain the above copyright
8\    notice, this list of conditions and the following disclaimer.
9\ 2. Redistributions in binary form must reproduce the above copyright
10\    notice, this list of conditions and the following disclaimer in the
11\    documentation and/or other materials provided with the distribution.
12\
13\ THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
14\ ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15\ IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16\ ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
17\ FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18\ DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19\ OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20\ HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21\ LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22\ OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
23\ SUCH DAMAGE.
24\
25\ $FreeBSD$
26
27\ Loader.rc support functions:
28\
29\ initialize ( addr len -- )	as above, plus load_conf_files
30\ load_conf ( addr len -- )	load conf file given
31\ include_conf_files ( -- )	load all conf files in load_conf_files
32\ print_syntax_error ( -- )	print line and marker of where a syntax
33\				error was detected
34\ print_line ( -- )		print last line processed
35\ load_kernel ( -- )		load kernel
36\ load_modules ( -- )		load modules flagged
37\
38\ Exported structures:
39\
40\ string			counted string structure
41\	cell .addr			string address
42\	cell .len			string length
43\ module			module loading information structure
44\	cell module.flag		should we load it?
45\	string module.name		module's name
46\	string module.loadname		name to be used in loading the module
47\	string module.type		module's type
48\	string module.args		flags to be passed during load
49\	string module.beforeload	command to be executed before load
50\	string module.afterload		command to be executed after load
51\	string module.loaderror		command to be executed if load fails
52\	cell module.next		list chain
53\
54\ Exported global variables;
55\
56\ string conf_files		configuration files to be loaded
57\ cell modules_options		pointer to first module information
58\ value verbose?		indicates if user wants a verbose loading
59\ value any_conf_read?		indicates if a conf file was successfully read
60\
61\ Other exported words:
62\    note, strlen is internal
63\ strdup ( addr len -- addr' len)			similar to strdup(3)
64\ strcat ( addr len addr' len' -- addr len+len' )	similar to strcat(3)
65\ s' ( | string' -- addr len | )			similar to s"
66\ rudimentary structure support
67
68\ Exception values
69
701 constant ESYNTAX
712 constant ENOMEM
723 constant EFREE
734 constant ESETERROR	\ error setting environment variable
745 constant EREAD	\ error reading
756 constant EOPEN
767 constant EEXEC	\ XXX never catched
778 constant EBEFORELOAD
789 constant EAFTERLOAD
79
80\ I/O constants
81
820 constant SEEK_SET
831 constant SEEK_CUR
842 constant SEEK_END
85
860 constant O_RDONLY
871 constant O_WRONLY
882 constant O_RDWR
89
90\ Crude structure support
91
92: structure:
93  create here 0 , ['] drop , 0
94  does> create here swap dup @ allot cell+ @ execute
95;
96: member: create dup , over , + does> cell+ @ + ;
97: ;structure swap ! ;
98: constructor! >body cell+ ! ;
99: constructor: over :noname ;
100: ;constructor postpone ; swap cell+ ! ; immediate
101: sizeof ' >body @ state @ if postpone literal then ; immediate
102: offsetof ' >body cell+ @ state @ if postpone literal then ; immediate
103: ptr 1 cells member: ;
104: int 1 cells member: ;
105
106\ String structure
107
108structure: string
109	ptr .addr
110	int .len
111	constructor:
112	  0 over .addr !
113	  0 swap .len !
114	;constructor
115;structure
116
117
118\ Module options linked list
119
120structure: module
121	int module.flag
122	sizeof string member: module.name
123	sizeof string member: module.loadname
124	sizeof string member: module.type
125	sizeof string member: module.args
126	sizeof string member: module.beforeload
127	sizeof string member: module.afterload
128	sizeof string member: module.loaderror
129	ptr module.next
130;structure
131
132\ Internal loader structures (preloaded_file, kernel_module, file_metadata)
133\ must be in sync with the C struct in stand/common/bootstrap.h
134structure: preloaded_file
135	ptr pf.name
136	ptr pf.type
137	ptr pf.args
138	ptr pf.metadata	\ file_metadata
139	int pf.loader
140	int pf.addr
141	int pf.size
142	ptr pf.modules	\ kernel_module
143	ptr pf.next	\ preloaded_file
144;structure
145
146structure: kernel_module
147	ptr km.name
148	\ ptr km.args
149	ptr km.fp	\ preloaded_file
150	ptr km.next	\ kernel_module
151;structure
152
153structure: file_metadata
154	int		md.size
155	2 member:	md.type	\ this is not ANS Forth compatible (XXX)
156	ptr		md.next	\ file_metadata
157	0 member:	md.data	\ variable size
158;structure
159
160\ end of structures
161
162\ Global variables
163
164string conf_files
165string nextboot_conf_file
166create module_options sizeof module.next allot 0 module_options !
167create last_module_option sizeof module.next allot 0 last_module_option !
1680 value verbose?
1690 value nextboot?
170
171\ Support string functions
172: strdup { addr len -- addr' len' }
173  len allocate if ENOMEM throw then
174  addr over len move len
175;
176
177: strcat  { addr len addr' len' -- addr len+len' }
178  addr' addr len + len' move
179  addr len len' +
180;
181
182: strchr { addr len c -- addr' len' }
183  begin
184    len
185  while
186    addr c@ c = if addr len exit then
187    addr 1 + to addr
188    len 1 - to len
189  repeat
190  0 0
191;
192
193: strspn { addr len addr1 len1 | paddr plen -- addr' len' }
194  begin
195    len
196  while
197    addr1 to paddr
198    len1 to plen
199    begin
200       plen
201    while
202       addr c@ paddr c@ = if addr len exit then
203       paddr 1+ to paddr
204       plen 1- to plen
205    repeat
206    addr 1 + to addr
207    len 1 - to len
208  repeat
209  0 0
210;
211
212: s' \ same as s", allows " in the string
213  [char] ' parse
214  state @ if postpone sliteral then
215; immediate
216
217: 2>r postpone >r postpone >r ; immediate
218: 2r> postpone r> postpone r> ; immediate
219: 2r@ postpone 2r> postpone 2dup postpone 2>r ; immediate
220
221: getenv?  getenv -1 = if false else drop true then ;
222
223\ execute xt for each device listed in console variable.
224\ this allows us to have device specific output for logos, menu frames etc
225: console-iterate { xt | caddr clen taddr tlen -- }
226	\ get current console and save it
227	s" console" getenv
228	['] strdup catch if 2drop exit then
229	to clen to caddr
230
231	clen to tlen
232	caddr to taddr
233	begin
234		tlen
235	while
236		taddr tlen s" , " strspn
237		\ we need to handle 3 cases for addr len pairs on stack:
238		\ addr len are 0 0 - there was no comma nor space
239		\ addr len are x 0 - the first char is either comma or space
240		\ addr len are x y.
241		2dup + 0= if
242			\ there was no comma nor space.
243			2drop
244			taddr tlen s" console" setenv
245			xt execute
246			0 to tlen
247		else dup 0= if
248			2drop
249		else
250			dup                     ( taddr' tlen' tlen' )
251			tlen swap - dup
252			0= if			\ sequence of comma and space?
253				drop
254			else
255				taddr swap s" console" setenv
256				xt execute
257			then
258			to tlen
259			to taddr
260		then then
261		tlen 0> if			\ step over separator
262			tlen 1- to tlen
263			taddr 1+ to taddr
264		then
265	repeat
266	caddr clen s" console" setenv		\ restore console setup
267	caddr free drop
268;
269
270\ determine if a word appears in a string, case-insensitive
271: contains? ( addr1 len1 addr2 len2 -- 0 | -1 )
272	2 pick 0= if 2drop 2drop true exit then
273	dup 0= if 2drop 2drop false exit then
274	begin
275		begin
276			swap dup c@ dup 32 = over 9 = or over 10 = or
277			over 13 = or over 44 = or swap drop
278		while 1+ swap 1- repeat
279		swap 2 pick 1- over <
280	while
281		2over 2over drop over compare-insensitive 0= if
282			2 pick over = if 2drop 2drop true exit then
283			2 pick tuck - -rot + swap over c@ dup 32 =
284			over 9 = or over 10 = or over 13 = or over 44 = or
285			swap drop if 2drop 2drop true exit then
286		then begin
287			swap dup c@ dup 32 = over 9 = or over 10 = or
288			over 13 = or over 44 = or swap drop
289			if false else true then 2 pick 0> and
290		while 1+ swap 1- repeat
291		swap
292	repeat
293	2drop 2drop false
294;
295
296: boot_serial? ( -- 0 | -1 )
297	s" console" getenv dup -1 <> if
298		s" comconsole" 2swap contains?
299	else drop false then
300\	s" boot_serial" getenv dup -1 <> if
301\		swap drop 0>
302\	else drop false then
303\	or \ console contains comconsole ( or ) boot_serial
304\	s" boot_multicons" getenv dup -1 <> if
305\		swap drop 0>
306\	else drop false then
307\	or \ previous boolean ( or ) boot_multicons
308;
309
310: framebuffer? ( -- t )
311	s" console" getenv
312	2dup s" efi" compare 0<> >r
313	s" vidconsole" compare 0<> r> and if
314		FALSE exit
315	then
316	s" screen.depth" getenv?
317;
318
319\ Private definitions
320
321vocabulary support-functions
322only forth also support-functions definitions
323
324\ Some control characters constants
325
3267 constant bell
3278 constant backspace
3289 constant tab
32910 constant lf
33013 constant <cr>
331
332\ Read buffer size
333
33480 constant read_buffer_size
335
336\ Standard suffixes
337
338: load_module_suffix		s" _load" ;
339: module_loadname_suffix	s" _name" ;
340: module_type_suffix		s" _type" ;
341: module_args_suffix		s" _flags" ;
342: module_beforeload_suffix	s" _before" ;
343: module_afterload_suffix	s" _after" ;
344: module_loaderror_suffix	s" _error" ;
345
346\ Support operators
347
348: >= < 0= ;
349: <= > 0= ;
350
351\ Assorted support functions
352
353: free-memory free if EFREE throw then ;
354
355: strget { var -- addr len } var .addr @ var .len @ ;
356
357\ assign addr len to variable.
358: strset  { addr len var -- } addr var .addr !  len var .len !  ;
359
360\ free memory and reset fields
361: strfree { var -- } var .addr @ ?dup if free-memory 0 0 var strset then ;
362
363\ free old content, make a copy of the string and assign to variable
364: string= { addr len var -- } var strfree addr len strdup var strset ;
365
366: strtype ( str -- ) strget type ;
367
368\ assign a reference to what is on the stack
369: strref { addr len var -- addr len }
370  addr var .addr ! len var .len ! addr len
371;
372
373\ unquote a string
374: unquote ( addr len -- addr len )
375  over c@ [char] " = if 2 chars - swap char+ swap then
376;
377
378\ Assignment data temporary storage
379
380string name_buffer
381string value_buffer
382
383\ Line by line file reading functions
384\
385\ exported:
386\	line_buffer
387\	end_of_file?
388\	fd
389\	read_line
390\	reset_line_reading
391
392vocabulary line-reading
393also line-reading definitions
394
395\ File data temporary storage
396
397string read_buffer
3980 value read_buffer_ptr
399
400\ File's line reading function
401
402get-current ( -- wid ) previous definitions
403
404string line_buffer
4050 value end_of_file?
406variable fd
407
408>search ( wid -- ) definitions
409
410: skip_newlines
411  begin
412    read_buffer .len @ read_buffer_ptr >
413  while
414    read_buffer .addr @ read_buffer_ptr + c@ lf = if
415      read_buffer_ptr char+ to read_buffer_ptr
416    else
417      exit
418    then
419  repeat
420;
421
422: scan_buffer  ( -- addr len )
423  read_buffer_ptr >r
424  begin
425    read_buffer .len @ r@ >
426  while
427    read_buffer .addr @ r@ + c@ lf = if
428      read_buffer .addr @ read_buffer_ptr +  ( -- addr )
429      r@ read_buffer_ptr -                   ( -- len )
430      r> to read_buffer_ptr
431      exit
432    then
433    r> char+ >r
434  repeat
435  read_buffer .addr @ read_buffer_ptr +  ( -- addr )
436  r@ read_buffer_ptr -                   ( -- len )
437  r> to read_buffer_ptr
438;
439
440: line_buffer_resize  ( len -- len )
441  dup 0= if exit then
442  >r
443  line_buffer .len @ if
444    line_buffer .addr @
445    line_buffer .len @ r@ +
446    resize if ENOMEM throw then
447  else
448    r@ allocate if ENOMEM throw then
449  then
450  line_buffer .addr !
451  r>
452;
453
454: append_to_line_buffer  ( addr len -- )
455  dup 0= if 2drop exit then
456  line_buffer strget
457  2swap strcat
458  line_buffer .len !
459  drop
460;
461
462: read_from_buffer
463  scan_buffer            ( -- addr len )
464  line_buffer_resize     ( len -- len )
465  append_to_line_buffer  ( addr len -- )
466;
467
468: refill_required?
469  read_buffer .len @ read_buffer_ptr =
470  end_of_file? 0= and
471;
472
473: refill_buffer
474  0 to read_buffer_ptr
475  read_buffer .addr @ 0= if
476    read_buffer_size allocate if ENOMEM throw then
477    read_buffer .addr !
478  then
479  fd @ read_buffer .addr @ read_buffer_size fread
480  dup -1 = if EREAD throw then
481  dup 0= if true to end_of_file? then
482  read_buffer .len !
483;
484
485get-current ( -- wid ) previous definitions >search ( wid -- )
486
487: reset_line_reading
488  0 to read_buffer_ptr
489;
490
491: read_line
492  line_buffer strfree
493  skip_newlines
494  begin
495    read_from_buffer
496    refill_required?
497  while
498    refill_buffer
499  repeat
500;
501
502only forth also support-functions definitions
503
504\ Conf file line parser:
505\ <line> ::= <spaces><name><spaces>'='<spaces><value><spaces>[<comment>] |
506\            <spaces>[<comment>]
507\ <name> ::= <letter>{<letter>|<digit>|'_'}
508\ <value> ::= '"'{<character_set>|'\'<anything>}'"' | <name>
509\ <character_set> ::= ASCII 32 to 126, except '\' and '"'
510\ <comment> ::= '#'{<anything>}
511\
512\ exported:
513\	line_pointer
514\	process_conf
515
5160 value line_pointer
517
518vocabulary file-processing
519also file-processing definitions
520
521\ parser functions
522\
523\ exported:
524\	get_assignment
525
526vocabulary parser
527also parser definitions
528
5290 value parsing_function
5300 value end_of_line
531
532: end_of_line?  line_pointer end_of_line = ;
533
534\ classifiers for various character classes in the input line
535
536: letter?
537  line_pointer c@ >r
538  r@ [char] A >=
539  r@ [char] Z <= and
540  r@ [char] a >=
541  r> [char] z <= and
542  or
543;
544
545: digit?
546  line_pointer c@ >r
547  r@ [char] - =
548  r@ [char] 0 >=
549  r> [char] 9 <= and
550  or
551;
552
553: quote?  line_pointer c@ [char] " = ;
554
555: assignment_sign?  line_pointer c@ [char] = = ;
556
557: comment?  line_pointer c@ [char] # = ;
558
559: space?  line_pointer c@ bl = line_pointer c@ tab = or ;
560
561: backslash?  line_pointer c@ [char] \ = ;
562
563: underscore?  line_pointer c@ [char] _ = ;
564
565: dot?  line_pointer c@ [char] . = ;
566
567\ manipulation of input line
568: skip_character line_pointer char+ to line_pointer ;
569
570: skip_to_end_of_line end_of_line to line_pointer ;
571
572: eat_space
573  begin
574    end_of_line? if 0 else space? then
575  while
576    skip_character
577  repeat
578;
579
580: parse_name  ( -- addr len )
581  line_pointer
582  begin
583    end_of_line? if 0 else letter? digit? underscore? dot? or or or then
584  while
585    skip_character
586  repeat
587  line_pointer over -
588  strdup
589;
590
591: remove_backslashes  { addr len | addr' len' -- addr' len' }
592  len allocate if ENOMEM throw then
593  to addr'
594  addr >r
595  begin
596    addr c@ [char] \ <> if
597      addr c@ addr' len' + c!
598      len' char+ to len'
599    then
600    addr char+ to addr
601    r@ len + addr =
602  until
603  r> drop
604  addr' len'
605;
606
607: parse_quote  ( -- addr len )
608  line_pointer
609  skip_character
610  end_of_line? if ESYNTAX throw then
611  begin
612    quote? 0=
613  while
614    backslash? if
615      skip_character
616      end_of_line? if ESYNTAX throw then
617    then
618    skip_character
619    end_of_line? if ESYNTAX throw then
620  repeat
621  skip_character
622  line_pointer over -
623  remove_backslashes
624;
625
626: read_name
627  parse_name		( -- addr len )
628  name_buffer strset
629;
630
631: read_value
632  quote? if
633    parse_quote		( -- addr len )
634  else
635    parse_name		( -- addr len )
636  then
637  value_buffer strset
638;
639
640: comment
641  skip_to_end_of_line
642;
643
644: white_space_4
645  eat_space
646  comment? if ['] comment to parsing_function exit then
647  end_of_line? 0= if ESYNTAX throw then
648;
649
650: variable_value
651  read_value
652  ['] white_space_4 to parsing_function
653;
654
655: white_space_3
656  eat_space
657  letter? digit? quote? or or if
658    ['] variable_value to parsing_function exit
659  then
660  ESYNTAX throw
661;
662
663: assignment_sign
664  skip_character
665  ['] white_space_3 to parsing_function
666;
667
668: white_space_2
669  eat_space
670  assignment_sign? if ['] assignment_sign to parsing_function exit then
671  ESYNTAX throw
672;
673
674: variable_name
675  read_name
676  ['] white_space_2 to parsing_function
677;
678
679: white_space_1
680  eat_space
681  letter?  if ['] variable_name to parsing_function exit then
682  comment? if ['] comment to parsing_function exit then
683  end_of_line? 0= if ESYNTAX throw then
684;
685
686get-current ( -- wid ) previous definitions >search ( wid -- )
687
688: get_assignment
689  line_buffer strget + to end_of_line
690  line_buffer .addr @ to line_pointer
691  ['] white_space_1 to parsing_function
692  begin
693    end_of_line? 0=
694  while
695    parsing_function execute
696  repeat
697  parsing_function ['] comment =
698  parsing_function ['] white_space_1 =
699  parsing_function ['] white_space_4 =
700  or or 0= if ESYNTAX throw then
701;
702
703only forth also support-functions also file-processing definitions
704
705\ Process line
706
707: assignment_type?  ( addr len -- flag )
708  name_buffer strget
709  compare 0=
710;
711
712: suffix_type?  ( addr len -- flag )
713  name_buffer .len @ over <= if 2drop false exit then
714  name_buffer .len @ over - name_buffer .addr @ +
715  over compare 0=
716;
717
718: loader_conf_files?  s" loader_conf_files" assignment_type?  ;
719
720: nextboot_flag?  s" nextboot_enable" assignment_type?  ;
721
722: nextboot_conf? s" nextboot_conf" assignment_type?  ;
723
724: verbose_flag? s" verbose_loading" assignment_type?  ;
725
726: execute? s" exec" assignment_type?  ;
727
728: module_load? load_module_suffix suffix_type? ;
729
730: module_loadname?  module_loadname_suffix suffix_type?  ;
731
732: module_type?  module_type_suffix suffix_type?  ;
733
734: module_args?  module_args_suffix suffix_type?  ;
735
736: module_beforeload?  module_beforeload_suffix suffix_type?  ;
737
738: module_afterload?  module_afterload_suffix suffix_type?  ;
739
740: module_loaderror?  module_loaderror_suffix suffix_type?  ;
741
742\ build a 'set' statement and execute it
743: set_environment_variable
744  name_buffer .len @ value_buffer .len @ + 5 chars + \ size of result string
745  allocate if ENOMEM throw then
746  dup 0  \ start with an empty string and append the pieces
747  s" set " strcat
748  name_buffer strget strcat
749  s" =" strcat
750  value_buffer strget strcat
751  ['] evaluate catch if
752    2drop free drop
753    ESETERROR throw
754  else
755    free-memory
756  then
757;
758
759: set_conf_files
760  set_environment_variable
761  s" loader_conf_files" getenv conf_files string=
762;
763
764: set_nextboot_conf
765  value_buffer strget unquote nextboot_conf_file string=
766;
767
768: append_to_module_options_list  ( addr -- )
769  module_options @ 0= if
770    dup module_options !
771    last_module_option !
772  else
773    dup last_module_option @ module.next !
774    last_module_option !
775  then
776;
777
778: set_module_name  { addr -- }	\ check leaks
779  name_buffer strget addr module.name string=
780;
781
782: yes_value?
783  value_buffer strget	\ XXX could use unquote
784  2dup s' "YES"' compare >r
785  2dup s' "yes"' compare >r
786  2dup s" YES" compare >r
787  s" yes" compare r> r> r> and and and 0=
788;
789
790: find_module_option  ( -- addr | 0 ) \ return ptr to entry matching name_buffer
791  module_options @
792  begin
793    dup
794  while
795    dup module.name strget
796    name_buffer strget
797    compare 0= if exit then
798    module.next @
799  repeat
800;
801
802: new_module_option  ( -- addr )
803  sizeof module allocate if ENOMEM throw then
804  dup sizeof module erase
805  dup append_to_module_options_list
806  dup set_module_name
807;
808
809: get_module_option  ( -- addr )
810  find_module_option
811  ?dup 0= if new_module_option then
812;
813
814: set_module_flag
815  name_buffer .len @ load_module_suffix nip - name_buffer .len !
816  yes_value? get_module_option module.flag !
817;
818
819: set_module_args
820  name_buffer .len @ module_args_suffix nip - name_buffer .len !
821  value_buffer strget unquote
822  get_module_option module.args string=
823;
824
825: set_module_loadname
826  name_buffer .len @ module_loadname_suffix nip - name_buffer .len !
827  value_buffer strget unquote
828  get_module_option module.loadname string=
829;
830
831: set_module_type
832  name_buffer .len @ module_type_suffix nip - name_buffer .len !
833  value_buffer strget unquote
834  get_module_option module.type string=
835;
836
837: set_module_beforeload
838  name_buffer .len @ module_beforeload_suffix nip - name_buffer .len !
839  value_buffer strget unquote
840  get_module_option module.beforeload string=
841;
842
843: set_module_afterload
844  name_buffer .len @ module_afterload_suffix nip - name_buffer .len !
845  value_buffer strget unquote
846  get_module_option module.afterload string=
847;
848
849: set_module_loaderror
850  name_buffer .len @ module_loaderror_suffix nip - name_buffer .len !
851  value_buffer strget unquote
852  get_module_option module.loaderror string=
853;
854
855: set_nextboot_flag
856  yes_value? to nextboot?
857;
858
859: set_verbose
860  yes_value? to verbose?
861;
862
863: execute_command
864  value_buffer strget unquote
865  ['] evaluate catch if EEXEC throw then
866;
867
868: process_assignment
869  name_buffer .len @ 0= if exit then
870  loader_conf_files?	if set_conf_files exit then
871  nextboot_flag?	if set_nextboot_flag exit then
872  nextboot_conf?	if set_nextboot_conf exit then
873  verbose_flag?		if set_verbose exit then
874  execute?		if execute_command exit then
875  module_load?		if set_module_flag exit then
876  module_loadname?	if set_module_loadname exit then
877  module_type?		if set_module_type exit then
878  module_args?		if set_module_args exit then
879  module_beforeload?	if set_module_beforeload exit then
880  module_afterload?	if set_module_afterload exit then
881  module_loaderror?	if set_module_loaderror exit then
882  set_environment_variable
883;
884
885\ free_buffer  ( -- )
886\
887\ Free some pointers if needed. The code then tests for errors
888\ in freeing, and throws an exception if needed. If a pointer is
889\ not allocated, it's value (0) is used as flag.
890
891: free_buffers
892  name_buffer strfree
893  value_buffer strfree
894;
895
896\ Higher level file processing
897
898get-current ( -- wid ) previous definitions >search ( wid -- )
899
900: process_conf
901  begin
902    end_of_file? 0=
903  while
904    free_buffers
905    read_line
906    get_assignment
907    ['] process_assignment catch
908    ['] free_buffers catch
909    swap throw throw
910  repeat
911;
912
913: peek_file ( addr len -- )
914  0 to end_of_file?
915  reset_line_reading
916  O_RDONLY fopen fd !
917  fd @ -1 = if EOPEN throw then
918  free_buffers
919  read_line
920  get_assignment
921  ['] process_assignment catch
922  ['] free_buffers catch
923  fd @ fclose
924  swap throw throw
925;
926
927only forth also support-functions definitions
928
929\ Interface to loading conf files
930
931: load_conf  ( addr len -- )
932  0 to end_of_file?
933  reset_line_reading
934  O_RDONLY fopen fd !
935  fd @ -1 = if EOPEN throw then
936  ['] process_conf catch
937  fd @ fclose
938  throw
939;
940
941: print_line line_buffer strtype cr ;
942
943: print_syntax_error
944  line_buffer strtype cr
945  line_buffer .addr @
946  begin
947    line_pointer over <>
948  while
949    bl emit char+
950  repeat
951  drop
952  ." ^" cr
953;
954
955
956\ Debugging support functions
957
958only forth definitions also support-functions
959
960: test-file
961  ['] load_conf catch dup .
962  ESYNTAX = if cr print_syntax_error then
963;
964
965\ find a module name, leave addr on the stack (0 if not found)
966: find-module ( <module> -- ptr | 0 )
967  bl parse ( addr len )
968  module_options @ >r ( store current pointer )
969  begin
970    r@
971  while
972    2dup ( addr len addr len )
973    r@ module.name strget
974    compare 0= if drop drop r> exit then ( found it )
975    r> module.next @ >r
976  repeat
977  type ."  was not found" cr r>
978;
979
980: show-nonempty ( addr len mod -- )
981  strget dup verbose? or if
982    2swap type type cr
983  else
984    drop drop drop drop
985  then ;
986
987: show-one-module { addr -- addr }
988  ." Name:        " addr module.name strtype cr
989  s" Path:        " addr module.loadname show-nonempty
990  s" Type:        " addr module.type show-nonempty
991  s" Flags:       " addr module.args show-nonempty
992  s" Before load: " addr module.beforeload show-nonempty
993  s" After load:  " addr module.afterload show-nonempty
994  s" Error:       " addr module.loaderror show-nonempty
995  ." Status:      " addr module.flag @ if ." Load" else ." Don't load" then cr
996  cr
997  addr
998;
999
1000: show-module-options
1001  module_options @
1002  begin
1003    ?dup
1004  while
1005    show-one-module
1006    module.next @
1007  repeat
1008;
1009
1010: free-one-module { addr -- addr }
1011  addr module.name strfree
1012  addr module.loadname strfree
1013  addr module.type strfree
1014  addr module.args strfree
1015  addr module.beforeload strfree
1016  addr module.afterload strfree
1017  addr module.loaderror strfree
1018  addr
1019;
1020
1021: free-module-options
1022  module_options @
1023  begin
1024    ?dup
1025  while
1026    free-one-module
1027    dup module.next @
1028    swap free-memory
1029  repeat
1030  0 module_options !
1031  0 last_module_option !
1032;
1033
1034only forth also support-functions definitions
1035
1036\ Variables used for processing multiple conf files
1037
1038string current_file_name_ref	\ used to print the file name
1039
1040\ Indicates if any conf file was successfully read
1041
10420 value any_conf_read?
1043
1044\ loader_conf_files processing support functions
1045
1046: get_conf_files ( -- addr len )  \ put addr/len on stack, reset var
1047  conf_files strget 0 0 conf_files strset
1048;
1049
1050: skip_leading_spaces  { addr len pos -- addr len pos' }
1051  begin
1052    pos len = if 0 else addr pos + c@ bl = then
1053  while
1054    pos char+ to pos
1055  repeat
1056  addr len pos
1057;
1058
1059\ return the file name at pos, or free the string if nothing left
1060: get_file_name  { addr len pos -- addr len pos' addr' len' || 0 }
1061  pos len = if
1062    addr free abort" Fatal error freeing memory"
1063    0 exit
1064  then
1065  pos >r
1066  begin
1067    \ stay in the loop until have chars and they are not blank
1068    pos len = if 0 else addr pos + c@ bl <> then
1069  while
1070    pos char+ to pos
1071  repeat
1072  addr len pos addr r@ + pos r> -
1073;
1074
1075: get_next_file  ( addr len ptr -- addr len ptr' addr' len' | 0 )
1076  skip_leading_spaces
1077  get_file_name
1078;
1079
1080: print_current_file
1081  current_file_name_ref strtype
1082;
1083
1084: process_conf_errors
1085  dup 0= if true to any_conf_read? drop exit then
1086  >r 2drop r>
1087  dup ESYNTAX = if
1088    ." Warning: syntax error on file " print_current_file cr
1089    print_syntax_error drop exit
1090  then
1091  dup ESETERROR = if
1092    ." Warning: bad definition on file " print_current_file cr
1093    print_line drop exit
1094  then
1095  dup EREAD = if
1096    ." Warning: error reading file " print_current_file cr drop exit
1097  then
1098  dup EOPEN = if
1099    verbose? if ." Warning: unable to open file " print_current_file cr then
1100    drop exit
1101  then
1102  dup EFREE = abort" Fatal error freeing memory"
1103  dup ENOMEM = abort" Out of memory"
1104  throw  \ Unknown error -- pass ahead
1105;
1106
1107\ Process loader_conf_files recursively
1108\ Interface to loader_conf_files processing
1109
1110: include_conf_files
1111  get_conf_files 0	( addr len offset )
1112  begin
1113    get_next_file ?dup ( addr len 1 | 0 )
1114  while
1115    current_file_name_ref strref
1116    ['] load_conf catch
1117    process_conf_errors
1118    conf_files .addr @ if recurse then
1119  repeat
1120;
1121
1122: get_nextboot_conf_file ( -- addr len )
1123  nextboot_conf_file strget
1124;
1125
1126: rewrite_nextboot_file ( -- )
1127  get_nextboot_conf_file
1128  O_WRONLY fopen fd !
1129  fd @ -1 = if EOPEN throw then
1130  fd @ s' nextboot_enable="NO" ' fwrite ( fd buf len -- nwritten ) drop
1131  fd @ fclose
1132;
1133
1134: include_nextboot_file ( -- )
1135  s" nextboot_enable" getenv dup -1 <> if
1136    2dup s' "YES"' compare >r
1137    2dup s' "yes"' compare >r
1138    2dup s" YES" compare >r
1139    2dup s" yes" compare r> r> r> and and and 0= to nextboot?
1140  else
1141    drop
1142    get_nextboot_conf_file
1143    ['] peek_file catch if 2drop then
1144  then
1145  nextboot? if
1146    get_nextboot_conf_file
1147    current_file_name_ref strref
1148    ['] load_conf catch
1149    process_conf_errors
1150    ['] rewrite_nextboot_file catch if 2drop then
1151  then
1152  s' "NO"' s" nextboot_enable" setenv
1153;
1154
1155\ Module loading functions
1156
1157: load_parameters  { addr -- addr addrN lenN ... addr1 len1 N }
1158  addr
1159  addr module.args strget
1160  addr module.loadname .len @ if
1161    addr module.loadname strget
1162  else
1163    addr module.name strget
1164  then
1165  addr module.type .len @ if
1166    addr module.type strget
1167    s" -t "
1168    4 ( -t type name flags )
1169  else
1170    2 ( name flags )
1171  then
1172;
1173
1174: before_load  ( addr -- addr )
1175  dup module.beforeload .len @ if
1176    dup module.beforeload strget
1177    ['] evaluate catch if EBEFORELOAD throw then
1178  then
1179;
1180
1181: after_load  ( addr -- addr )
1182  dup module.afterload .len @ if
1183    dup module.afterload strget
1184    ['] evaluate catch if EAFTERLOAD throw then
1185  then
1186;
1187
1188: load_error  ( addr -- addr )
1189  dup module.loaderror .len @ if
1190    dup module.loaderror strget
1191    evaluate  \ This we do not intercept so it can throw errors
1192  then
1193;
1194
1195: pre_load_message  ( addr -- addr )
1196  verbose? if
1197    dup module.name strtype
1198    ." ..."
1199  then
1200;
1201
1202: load_error_message verbose? if ." failed!" cr then ;
1203
1204: load_successful_message verbose? if ." ok" cr then ;
1205
1206: load_module
1207  load_parameters load
1208;
1209
1210: process_module  ( addr -- addr )
1211  pre_load_message
1212  before_load
1213  begin
1214    ['] load_module catch if
1215      dup module.loaderror .len @ if
1216        load_error			\ Command should return a flag!
1217      else
1218        load_error_message true		\ Do not retry
1219      then
1220    else
1221      after_load
1222      load_successful_message true	\ Successful, do not retry
1223    then
1224  until
1225;
1226
1227: process_module_errors  ( addr ior -- )
1228  dup EBEFORELOAD = if
1229    drop
1230    ." Module "
1231    dup module.name strtype
1232    dup module.loadname .len @ if
1233      ." (" dup module.loadname strtype ." )"
1234    then
1235    cr
1236    ." Error executing "
1237    dup module.beforeload strtype cr	\ XXX there was a typo here
1238    abort
1239  then
1240
1241  dup EAFTERLOAD = if
1242    drop
1243    ." Module "
1244    dup module.name .addr @ over module.name .len @ type
1245    dup module.loadname .len @ if
1246      ." (" dup module.loadname strtype ." )"
1247    then
1248    cr
1249    ." Error executing "
1250    dup module.afterload strtype cr
1251    abort
1252  then
1253
1254  throw  \ Don't know what it is all about -- pass ahead
1255;
1256
1257\ Module loading interface
1258
1259\ scan the list of modules, load enabled ones.
1260: load_modules  ( -- ) ( throws: abort & user-defined )
1261  module_options @	( list_head )
1262  begin
1263    ?dup
1264  while
1265    dup module.flag @ if
1266      ['] process_module catch
1267      process_module_errors
1268    then
1269    module.next @
1270  repeat
1271;
1272
1273\ h00h00 magic used to try loading either a kernel with a given name,
1274\ or a kernel with the default name in a directory of a given name
1275\ (the pain!)
1276
1277: bootpath s" /boot/" ;
1278: modulepath s" module_path" ;
1279
1280\ Functions used to save and restore module_path's value.
1281: saveenv ( addr len | -1 -- addr' len | 0 -1 )
1282  dup -1 = if 0 swap exit then
1283  strdup
1284;
1285: freeenv ( addr len | 0 -1 )
1286  -1 = if drop else free abort" Freeing error" then
1287;
1288: restoreenv  ( addr len | 0 -1 -- )
1289  dup -1 = if ( it wasn't set )
1290    2drop
1291    modulepath unsetenv
1292  else
1293    over >r
1294    modulepath setenv
1295    r> free abort" Freeing error"
1296  then
1297;
1298
1299: clip_args   \ Drop second string if only one argument is passed
1300  1 = if
1301    2swap 2drop
1302    1
1303  else
1304    2
1305  then
1306;
1307
1308also builtins
1309
1310\ Parse filename from a semicolon-separated list
1311
1312\ replacement, not working yet
1313: newparse-; { addr len | a1 -- a' len-x addr x }
1314  addr len [char] ; strchr dup if	( a1 len1 )
1315    swap to a1	( store address )
1316    1 - a1 @ 1 + swap ( remove match )
1317    addr a1 addr -
1318  else
1319    0 0 addr len
1320  then
1321;
1322
1323: parse-; ( addr len -- addr' len-x addr x )
1324  over 0 2swap			( addr 0 addr len )
1325  begin
1326    dup 0 <>			( addr 0 addr len )
1327  while
1328    over c@ [char] ; <>		( addr 0 addr len flag )
1329  while
1330    1- swap 1+ swap
1331    2swap 1+ 2swap
1332  repeat then
1333  dup 0 <> if
1334    1- swap 1+ swap
1335  then
1336  2swap
1337;
1338
1339\ Try loading one of multiple kernels specified
1340
1341: try_multiple_kernels ( addr len addr' len' args -- flag )
1342  >r
1343  begin
1344    parse-; 2>r
1345    2over 2r>
1346    r@ clip_args
1347    s" DEBUG" getenv? if
1348      s" echo Module_path: ${module_path}" evaluate
1349      ." Kernel     : " >r 2dup type r> cr
1350      dup 2 = if ." Flags      : " >r 2over type r> cr then
1351    then
1352    1 load
1353  while
1354    dup 0=
1355  until
1356    1 >r \ Failure
1357  else
1358    0 >r \ Success
1359  then
1360  2drop 2drop
1361  r>
1362  r> drop
1363;
1364
1365\ Try to load a kernel; the kernel name is taken from one of
1366\ the following lists, as ordered:
1367\
1368\   1. The "bootfile" environment variable
1369\   2. The "kernel" environment variable
1370\
1371\ Flags are passed, if available. If not, dummy values must be given.
1372\
1373\ The kernel gets loaded from the current module_path.
1374
1375: load_a_kernel ( flags len 1 | x x 0 -- flag )
1376  local args
1377  2local flags
1378  0 0 2local kernel
1379  end-locals
1380
1381  \ Check if a default kernel name exists at all, exits if not
1382  s" bootfile" getenv dup -1 <> if
1383    to kernel
1384    flags kernel args 1+ try_multiple_kernels
1385    dup 0= if exit then
1386  then
1387  drop
1388
1389  s" kernel" getenv dup -1 <> if
1390    to kernel
1391  else
1392    drop
1393    1 exit \ Failure
1394  then
1395
1396  \ Try all default kernel names
1397  flags kernel args 1+ try_multiple_kernels
1398;
1399
1400\ Try to load a kernel; the kernel name is taken from one of
1401\ the following lists, as ordered:
1402\
1403\   1. The "bootfile" environment variable
1404\   2. The "kernel" environment variable
1405\
1406\ Flags are passed, if provided.
1407\
1408\ The kernel will be loaded from a directory computed from the
1409\ path given. Two directories will be tried in the following order:
1410\
1411\   1. /boot/path
1412\   2. path
1413\
1414\ The module_path variable is overridden if load is successful, by
1415\ prepending the successful path.
1416
1417: load_from_directory ( path len 1 | flags len' path len 2 -- flag )
1418  local args
1419  2local path
1420  args 1 = if 0 0 then
1421  2local flags
1422  0 0 2local oldmodulepath \ like a string
1423  0 0 2local newmodulepath \ like a string
1424  end-locals
1425
1426  \ Set the environment variable module_path, and try loading
1427  \ the kernel again.
1428  modulepath getenv saveenv to oldmodulepath
1429
1430  \ Try prepending /boot/ first
1431  bootpath nip path nip + 	\ total length
1432  oldmodulepath nip dup -1 = if
1433    drop
1434  else
1435    1+ +			\ add oldpath -- XXX why the 1+ ?
1436  then
1437  allocate if ( out of memory ) 1 exit then \ XXX throw ?
1438
1439  0
1440  bootpath strcat
1441  path strcat
1442  2dup to newmodulepath
1443  modulepath setenv
1444
1445  \ Try all default kernel names
1446  flags args 1- load_a_kernel
1447  0= if ( success )
1448    oldmodulepath nip -1 <> if
1449      newmodulepath s" ;" strcat
1450      oldmodulepath strcat
1451      modulepath setenv
1452      newmodulepath drop free-memory
1453      oldmodulepath drop free-memory
1454    then
1455    0 exit
1456  then
1457
1458  \ Well, try without the prepended /boot/
1459  path newmodulepath drop swap move
1460  newmodulepath drop path nip
1461  2dup to newmodulepath
1462  modulepath setenv
1463
1464  \ Try all default kernel names
1465  flags args 1- load_a_kernel
1466  if ( failed once more )
1467    oldmodulepath restoreenv
1468    newmodulepath drop free-memory
1469    1
1470  else
1471    oldmodulepath nip -1 <> if
1472      newmodulepath s" ;" strcat
1473      oldmodulepath strcat
1474      modulepath setenv
1475      newmodulepath drop free-memory
1476      oldmodulepath drop free-memory
1477    then
1478    0
1479  then
1480;
1481
1482\ Try to load a kernel; the kernel name is taken from one of
1483\ the following lists, as ordered:
1484\
1485\   1. The "bootfile" environment variable
1486\   2. The "kernel" environment variable
1487\   3. The "path" argument
1488\
1489\ Flags are passed, if provided.
1490\
1491\ The kernel will be loaded from a directory computed from the
1492\ path given. Two directories will be tried in the following order:
1493\
1494\   1. /boot/path
1495\   2. path
1496\
1497\ Unless "path" is meant to be kernel name itself. In that case, it
1498\ will first be tried as a full path, and, next, search on the
1499\ directories pointed by module_path.
1500\
1501\ The module_path variable is overridden if load is successful, by
1502\ prepending the successful path.
1503
1504: load_directory_or_file ( path len 1 | flags len' path len 2 -- flag )
1505  local args
1506  2local path
1507  args 1 = if 0 0 then
1508  2local flags
1509  end-locals
1510
1511  \ First, assume path is an absolute path to a directory
1512  flags path args clip_args load_from_directory
1513  dup 0= if exit else drop then
1514
1515  \ Next, assume path points to the kernel
1516  flags path args try_multiple_kernels
1517;
1518
1519: initialize  ( addr len -- )
1520  strdup conf_files strset
1521;
1522
1523: kernel_options ( -- addr len 1 | 0 )
1524  s" kernel_options" getenv
1525  dup -1 = if drop 0 else 1 then
1526;
1527
1528: standard_kernel_search  ( flags 1 | 0 -- flag )
1529  local args
1530  args 0= if 0 0 then
1531  2local flags
1532  s" kernel" getenv
1533  dup -1 = if 0 swap then
1534  2local path
1535  end-locals
1536
1537  path nip -1 = if ( there isn't a "kernel" environment variable )
1538    flags args load_a_kernel
1539  else
1540    flags path args 1+ clip_args load_directory_or_file
1541  then
1542;
1543
1544: load_kernel  ( -- ) ( throws: abort )
1545  kernel_options standard_kernel_search
1546  abort" Unable to load a kernel!"
1547;
1548
1549: load_xen ( -- flag )
1550  s" xen_kernel" getenv dup -1 <> if
1551    1 1 load ( c-addr/u flag N -- flag )
1552  else
1553    drop
1554    0 ( -1 -- flag )
1555  then
1556;
1557
1558: load_xen_throw ( -- ) ( throws: abort )
1559  load_xen
1560  abort" Unable to load Xen!"
1561;
1562
1563: set_defaultoptions  ( -- )
1564  s" kernel_options" getenv dup -1 = if
1565    drop
1566  else
1567    s" temp_options" setenv
1568  then
1569;
1570
1571\ pick the i-th argument, i starts at 0
1572: argv[]  ( aN uN ... a1 u1 N i -- aN uN ... a1 u1 N ai+1 ui+1 )
1573  2dup = if 0 0 exit then	\ out of range
1574  dup >r
1575  1+ 2* ( skip N and ui )
1576  pick
1577  r>
1578  1+ 2* ( skip N and ai )
1579  pick
1580;
1581
1582: drop_args  ( aN uN ... a1 u1 N -- )
1583  0 ?do 2drop loop
1584;
1585
1586: argc
1587  dup
1588;
1589
1590: queue_argv  ( aN uN ... a1 u1 N a u -- a u aN uN ... a1 u1 N+1 )
1591  >r
1592  over 2* 1+ -roll
1593  r>
1594  over 2* 1+ -roll
1595  1+
1596;
1597
1598: unqueue_argv  ( aN uN ... a1 u1 N -- aN uN ... a2 u2 N-1 a1 u1 )
1599  1- -rot
1600;
1601
1602\ compute the length of the buffer including the spaces between words
1603: strlen(argv) ( aN uN .. a1 u1 N -- aN uN .. a1 u1 N len )
1604  dup 0= if 0 exit then
1605  0 >r	\ Size
1606  0 >r	\ Index
1607  begin
1608    argc r@ <>
1609  while
1610    r@ argv[]
1611    nip
1612    r> r> rot + 1+
1613    >r 1+ >r
1614  repeat
1615  r> drop
1616  r>
1617;
1618
1619: concat_argv  ( aN uN ... a1 u1 N -- a u )
1620  strlen(argv) allocate if ENOMEM throw then
1621  0 2>r ( save addr 0 on return stack )
1622
1623  begin
1624    dup
1625  while
1626    unqueue_argv ( ... N a1 u1 )
1627    2r> 2swap	 ( old a1 u1 )
1628    strcat
1629    s"  " strcat ( append one space ) \ XXX this gives a trailing space
1630    2>r		( store string on the result stack )
1631  repeat
1632  drop_args
1633  2r>
1634;
1635
1636: set_tempoptions  ( addrN lenN ... addr1 len1 N -- addr len 1 | 0 )
1637  \ Save the first argument, if it exists and is not a flag
1638  argc if
1639    0 argv[] drop c@ [char] - <> if
1640      unqueue_argv 2>r  \ Filename
1641      1 >r		\ Filename present
1642    else
1643      0 >r		\ Filename not present
1644    then
1645  else
1646    0 >r		\ Filename not present
1647  then
1648
1649  \ If there are other arguments, assume they are flags
1650  ?dup if
1651    concat_argv
1652    2dup s" temp_options" setenv
1653    drop free if EFREE throw then
1654  else
1655    set_defaultoptions
1656  then
1657
1658  \ Bring back the filename, if one was provided
1659  r> if 2r> 1 else 0 then
1660;
1661
1662: get_arguments ( -- addrN lenN ... addr1 len1 N )
1663  0
1664  begin
1665    \ Get next word on the command line
1666    parse-word
1667  ?dup while
1668    queue_argv
1669  repeat
1670  drop ( empty string )
1671;
1672
1673: load_kernel_and_modules  ( args -- flag )
1674  set_tempoptions
1675  argc >r
1676  s" temp_options" getenv dup -1 <> if
1677    queue_argv
1678  else
1679    drop
1680  then
1681  load_xen
1682  ?dup 0= if ( success )
1683    r> if ( a path was passed )
1684      load_directory_or_file
1685    else
1686      standard_kernel_search
1687    then
1688    ?dup 0= if ['] load_modules catch then
1689  then
1690;
1691
1692only forth definitions
1693