1
0
Fork 0
mirror of https://github.com/ganelson/inform.git synced 2024-07-02 23:14:57 +03:00
inform7/inbuild/Chapter 1/Main.w

223 lines
7.2 KiB
OpenEdge ABL
Raw Normal View History

2020-01-28 00:51:46 +02:00
[Main::] Main.
The top level, which decides what is to be done and then carries
this plan out.
@h Main routine.
@d INTOOL_NAME "inbuild"
@e INSPECT_TTASK from 1
@e GRAPH_TTASK
@e BUILD_TTASK
@e REBUILD_TTASK
2020-02-01 01:04:39 +02:00
@e COPY_TO_TTASK
@e SYNC_TO_TTASK
2020-01-28 00:51:46 +02:00
=
pathname *path_to_inbuild = NULL;
pathname *path_to_tools = NULL;
2020-01-31 02:13:50 +02:00
linked_list *nest_list = NULL;
linked_list *find_list = NULL;
int inbuild_task = INSPECT_TTASK;
int dry_run_mode = FALSE;
linked_list *targets = NULL; /* of |inbuild_copy| */
2020-02-01 01:04:39 +02:00
inbuild_nest *destination_nest = NULL;
2020-01-28 00:51:46 +02:00
int main(int argc, char **argv) {
Foundation::start();
WordsModule::start();
InbuildModule::start();
targets = NEW_LINKED_LIST(inbuild_copy);
2020-01-31 02:13:50 +02:00
nest_list = NEW_LINKED_LIST(inbuild_nest);
find_list = NEW_LINKED_LIST(text_stream);
2020-01-28 00:51:46 +02:00
@<Read the command line@>;
2020-01-31 02:13:50 +02:00
if (FIRST_IN_LINKED_LIST(inbuild_nest, nest_list) == NULL)
Nests::add_to_search_sequence(nest_list,
Nests::new(Pathnames::from_text(I"inform7/Internal")));
path_to_inbuild = Pathnames::installation_path("INBUILD_PATH", I"inbuild");
build_methodology *BM;
if (path_to_tools) BM = BuildSteps::methodology(path_to_tools, FALSE);
else BM = BuildSteps::methodology(Pathnames::up(path_to_inbuild), TRUE);
if (dry_run_mode == FALSE) BM->methodology = SHELL_METHODOLOGY;
2020-01-31 02:13:50 +02:00
text_stream *T;
LOOP_OVER_LINKED_LIST(T, text_stream, find_list) {
linked_list *L = NEW_LINKED_LIST(inbuild_search_result);
inbuild_work *work = Works::new(kit_genre, T, I"");
2020-01-31 02:13:50 +02:00
inbuild_requirement *req = Model::requirement(work,
VersionNumbers::null(), VersionNumbers::null());
Nests::locate(req, nest_list, L);
int n = 0;
inbuild_search_result *R;
LOOP_OVER_LINKED_LIST(R, inbuild_search_result, L) {
Model::write_copy(STDOUT, R->copy);
WRITE_TO(STDOUT, " (in nest %p)\n", R->nest->location);
n++;
}
if (n == 0) WRITE_TO(STDOUT, "%S not found\n", T);
}
inbuild_copy *C;
LOOP_OVER_LINKED_LIST(C, inbuild_copy, targets) {
switch (inbuild_task) {
case INSPECT_TTASK: Graphs::describe(STDOUT, C->graph, FALSE); break;
case GRAPH_TTASK: Graphs::describe(STDOUT, C->graph, TRUE); break;
case BUILD_TTASK: Graphs::build(C->graph, BM); break;
case REBUILD_TTASK: Graphs::rebuild(C->graph, BM); break;
2020-02-01 01:04:39 +02:00
case COPY_TO_TTASK: if (destination_nest) Nests::copy_to(C, destination_nest, FALSE); break;
case SYNC_TO_TTASK: if (destination_nest) Nests::copy_to(C, destination_nest, TRUE); break;
}
}
WordsModule::end();
InbuildModule::end();
2020-01-28 00:51:46 +02:00
Foundation::end();
return 0;
}
@ We use Foundation to read the command line:
@e BUILD_CLSW
@e REBUILD_CLSW
@e GRAPH_CLSW
@e INSPECT_CLSW
@e DRY_CLSW
@e TOOLS_CLSW
@e CONTENTS_OF_CLSW
2020-01-31 02:13:50 +02:00
@e NEST_CLSW
@e FIND_CLSW
2020-02-01 01:04:39 +02:00
@e COPY_TO_CLSW
@e SYNC_TO_CLSW
2020-01-28 00:51:46 +02:00
@<Read the command line@> =
CommandLine::declare_heading(
L"[[Purpose]]\n\n"
L"usage: inbuild [-TASK] TARGET1 TARGET2 ...\n");
2020-02-01 01:04:39 +02:00
CommandLine::declare_switch(COPY_TO_CLSW, L"copy-to", 2,
L"copy target(s) to nest X");
CommandLine::declare_switch(SYNC_TO_CLSW, L"sync-to", 2,
L"forcibly copy target(s) to nest X, even if prior version already there");
CommandLine::declare_switch(BUILD_CLSW, L"build", 1,
L"incrementally build target(s)");
CommandLine::declare_switch(REBUILD_CLSW, L"rebuild", 1,
L"completely rebuild target(s)");
CommandLine::declare_switch(INSPECT_CLSW, L"inspect", 1,
L"show target(s) but take no action");
CommandLine::declare_switch(GRAPH_CLSW, L"graph", 1,
L"show dependency graph of target(s) but take no action");
CommandLine::declare_switch(TOOLS_CLSW, L"tools", 2,
L"make X the directory of intools executables, and exit developer mode");
CommandLine::declare_boolean_switch(DRY_CLSW, L"dry", 1,
L"make this a dry run (print but do not execute shell commands)");
CommandLine::declare_boolean_switch(CONTENTS_OF_CLSW, L"contents-of", 2,
L"apply to all targets in the directory X");
2020-01-31 02:13:50 +02:00
CommandLine::declare_switch(NEST_CLSW, L"nest", 2,
L"add the nest at pathname X to the search list");
CommandLine::declare_switch(FIND_CLSW, L"find", 2,
L"find copies in nests in the search list");
2020-01-28 00:51:46 +02:00
CommandLine::read(argc, argv, NULL, &Main::option, &Main::bareword);
@ =
void Main::option(int id, int val, text_stream *arg, void *state) {
switch (id) {
case BUILD_CLSW: inbuild_task = BUILD_TTASK; break;
case REBUILD_CLSW: inbuild_task = REBUILD_TTASK; break;
case INSPECT_CLSW: inbuild_task = INSPECT_TTASK; break;
case GRAPH_CLSW: inbuild_task = GRAPH_TTASK; break;
case TOOLS_CLSW: path_to_tools = Pathnames::from_text(arg); break;
case CONTENTS_OF_CLSW: Main::load_many(Pathnames::from_text(arg)); break;
case DRY_CLSW: dry_run_mode = val; break;
2020-01-31 02:13:50 +02:00
case NEST_CLSW: Nests::add_to_search_sequence(nest_list,
Nests::new(Pathnames::from_text(arg))); break;
case FIND_CLSW: ADD_TO_LINKED_LIST(Str::duplicate(arg), text_stream, find_list); break;
2020-02-01 01:04:39 +02:00
case COPY_TO_CLSW: inbuild_task = COPY_TO_TTASK;
destination_nest = Nests::new(Pathnames::from_text(arg));
break;
case SYNC_TO_CLSW: inbuild_task = SYNC_TO_TTASK;
destination_nest = Nests::new(Pathnames::from_text(arg));
break;
default: internal_error("unimplemented switch");
}
2020-01-28 00:51:46 +02:00
}
void Main::bareword(int id, text_stream *arg, void *state) {
Main::load_one(arg, TRUE);
}
void Main::load_many(pathname *P) {
scan_directory *D = Directories::open(P);
TEMPORARY_TEXT(LEAFNAME);
while (Directories::next(D, LEAFNAME)) {
TEMPORARY_TEXT(FILENAME);
WRITE_TO(FILENAME, "%p%c%S", P, FOLDER_SEPARATOR, LEAFNAME);
Main::load_one(FILENAME, FALSE);
DISCARD_TEXT(FILENAME);
}
DISCARD_TEXT(LEAFNAME);
Directories::close(D);
}
void Main::load_one(text_stream *arg, int throwing_error) {
inbuild_copy *C = Model::claim(arg);
if (C == NULL) {
if (throwing_error) Errors::with_text("unable to identify '%S'", arg);
return;
}
ADD_TO_LINKED_LIST(C, inbuild_copy, targets);
2020-01-28 00:51:46 +02:00
}
@ Since we want to include the words module, we have to define the following
structure and initialiser:
@d VOCABULARY_MEANING_INITIALISER Main::ignore
=
typedef struct vocabulary_meaning {
int enigmatic_number;
} vocabulary_meaning;
@
@d LEXER_PROBLEM_HANDLER Main::lexer_problem_handler
=
vocabulary_meaning Main::ignore(vocabulary_entry *ve) {
vocabulary_meaning vm;
vm.enigmatic_number = 16339;
return vm;
}
void Main::lexer_problem_handler(int err, text_stream *problem_source_description, wchar_t *word) {
if (err == MEMORY_OUT_LEXERERROR)
Errors::fatal("Out of memory: unable to create lexer workspace");
TEMPORARY_TEXT(word_t);
if (word) WRITE_TO(word_t, "%w", word);
switch (err) {
case STRING_TOO_LONG_LEXERERROR:
Errors::with_text("Too much text in quotation marks: %S", word_t);
break;
case WORD_TOO_LONG_LEXERERROR:
Errors::with_text("Word too long: %S", word_t);
break;
case I6_TOO_LONG_LEXERERROR:
Errors::with_text("I6 inclusion too long: %S", word_t);
break;
case STRING_NEVER_ENDS_LEXERERROR:
Errors::with_text("Quoted text never ends: %S", problem_source_description);
break;
case COMMENT_NEVER_ENDS_LEXERERROR:
Errors::with_text("Square-bracketed text never ends: %S", problem_source_description);
break;
case I6_NEVER_ENDS_LEXERERROR:
Errors::with_text("I6 inclusion text never ends: %S", problem_source_description);
break;
default:
internal_error("unknown lexer error");
}
DISCARD_TEXT(word_t);
}
@
@d PREFORM_LANGUAGE_TYPE void