diff --git a/.gitignore b/.gitignore index be94c4455..4f79e95eb 100644 --- a/.gitignore +++ b/.gitignore @@ -56,4 +56,4 @@ gdbinit src/components/lib/libc/musl-1.2.0/include/asm/ src/components/lib/libc/musl-1.2.0/include/asm-generic/ src/components/lib/libc/musl-1.2.0/include/linux/ -src/components/lib/libc/musl-1.2.0/include/sys/queue.h \ No newline at end of file +src/components/lib/libc/musl-1.2.0/include/sys/queue.h diff --git a/composition_scripts/burn.toml b/composition_scripts/burn.toml new file mode 100644 index 000000000..5adeaa8af --- /dev/null +++ b/composition_scripts/burn.toml @@ -0,0 +1,15 @@ +[system] +description = "Simple system running only a CPU-burn component." + +[[components]] +name = "booter" +img = "no_interface.llbooter" +implements = [{interface = "init"}, {interface = "addr"}] +deps = [{srv = "kernel", interface = "init", variant = "kernel"}] +constructor = "kernel" + +[[components]] +name = "burn" +img = "no_interface.burn" +deps = [{srv = "booter", interface = "init"}, {srv = "booter", interface = "addr"}] +constructor = "booter" diff --git a/composition_scripts/external_hello_world.toml b/composition_scripts/external_hello_world.toml new file mode 100644 index 000000000..a5c81a96a --- /dev/null +++ b/composition_scripts/external_hello_world.toml @@ -0,0 +1,9 @@ +[system] +description = "Test for the hello world external component" + +[[components]] +name = "hw1" +img = "" +repo = "github:gparmer/hello_world" +deps = [{srv = "kernel", interface = "init", variant = "kernel"}] +constructor = "kernel" diff --git a/src/Makefile b/src/Makefile index 8ccee4cba..e683284f5 100644 --- a/src/Makefile +++ b/src/Makefile @@ -1,6 +1,7 @@ #MAKEFLAGS=--no-print-directory --section-alignment 0x1000 -I$(PWD) # Note: can't use $(PWD) here if make -Ced from elsewhere. MAKEOPTIONS=--no-print-directory -I$(shell pwd) +export MAKEOPTIONS PLAT_FILE=.PLATFORM_ID .PHONY: default all composer component comps platclean plat cpplat clean distclean init update config @@ -20,6 +21,10 @@ composer: component: $(MAKE) $(MAKEOPTIONS) -C components component +# Download an external repository +repo_external: + $(MAKE) $(MAKEOPTIONS) -C components repo_external + comps: $(info ) $(info ***********************************************) diff --git a/src/Makefile.src b/src/Makefile.src index 56badc076..4441e9cab 100644 --- a/src/Makefile.src +++ b/src/Makefile.src @@ -11,6 +11,7 @@ KINC=$(TOP_DIR)/kernel/include/ SHAREDINC=$(TOP_DIR)/kernel/include/shared/ CHALSHAREDINC=$(TOP_DIR)/kernel/include/chal/shared/ CDIR=$(TOP_DIR)/components/ +REPO_DIR=$(CDIR)/repos # tools MAKE=make @@ -25,3 +26,8 @@ CP=cp RM=rm PLAT_FILE=$(TOP_DIR)/.PLATFORM_ID PLATFORM=$(shell cat $(PLAT_FILE)) + +# disable the terminal prompts, and fail out instead; typically fails +# if ssh isn't set up (known hosts), or if the specified repo doesn't +# exist. +GITCLONE=GIT_TERMINAL_PROMPT=0 git clone diff --git a/src/components/Makefile b/src/components/Makefile index 9aa6d489f..014545ec2 100644 --- a/src/components/Makefile +++ b/src/components/Makefile @@ -1,4 +1,6 @@ -MAKEOPTIONS=-I$(shell pwd) +include Makefile.src +MAKEOPTIONS+= -I$(shell pwd) +export MAKEOPTIONS # Order of which subdirectories are built first matters here. # Interface files rely on library files, and component implementations @@ -24,3 +26,12 @@ init: component: $(MAKE) $(MAKEOPTIONS) -C implementation component + +# Download a remote repository (e.g. from github). We're setting up a +# specific rule for the repo's path so that it won't redundantly clone +# the repo. Once it has been cloned, never clone it again. +REPO_ABSPATH = $(REPO_DIR)/$(REPO_PATH) +$(REPO_ABSPATH): + $(GITCLONE) $(REPO_URL) $@ + +repo_external: $(REPO_ABSPATH) diff --git a/src/components/implementation/Makefile b/src/components/implementation/Makefile index 69ad4c1aa..75a36c563 100644 --- a/src/components/implementation/Makefile +++ b/src/components/implementation/Makefile @@ -1,7 +1,8 @@ include Makefile.src Makefile.comp SUBDIRS=$(filter-out skel/, $(wildcard */)) -MAKEOPTIONS=-I$(shell pwd) +MAKEOPTIONS+= -I$(shell pwd) +export MAKEOPTIONS .PHONY: all all: @@ -23,5 +24,12 @@ clean: init: distclean: +# NOTE: we're skipping the compilation of the libraries in the +# interfaces directories (the immediate subdirectories here) on +# composer-directed compilation. Soon, we'll remove the need for +# interface directories, and will move all of their shared code into +# the `lib/` code. +# +# NOTE: COMP_PATH is relative to the `src/component` (CDIR) directory. component: - $(MAKE) $(MAKEOPTIONS) -C $(COMP_INTERFACE) component + $(MAKE) $(MAKEOPTIONS) -C $(CDIR)/$(COMP_PATH) component diff --git a/src/components/implementation/Makefile.subdir b/src/components/implementation/Makefile.subdir index 659f4c340..a050ab81e 100644 --- a/src/components/implementation/Makefile.subdir +++ b/src/components/implementation/Makefile.subdir @@ -21,7 +21,7 @@ print: .PHONY: subdirs subdirs: @for dir in $(SUBDIRS) ; do \ - $(MAKE) -C $$dir ; \ + $(MAKE) $(MAKEOPTIONS) -C $$dir ; \ done $(CLIB):$(OBJS) @@ -75,6 +75,6 @@ clean: .PHONY: component component: - $(MAKE) -C $(COMP_NAME) component + $(MAKE) $(MAKEOPTIONS) -C $(COMP_NAME) component -include $(SOURCE_DEPENDENCIES) diff --git a/src/components/implementation/Makefile.subsubdir b/src/components/implementation/Makefile.subsubdir index 888f90991..494739d1f 100644 --- a/src/components/implementation/Makefile.subsubdir +++ b/src/components/implementation/Makefile.subsubdir @@ -40,7 +40,7 @@ ifeq ($(PLATFORM), i386) LINKFLAG=-Xlinker -melf_i386 COMP_LD_SCRIPT=$(IMPLDIR)/comp_i386.ld else ifeq ($(PLATFORM), x86_64) -LINKFLAG=-Xlinker -melf_x86_64 +LINKFLAG=-Xlinker -melf_x86_64 COMP_LD_SCRIPT=$(IMPLDIR)/comp_x86_64.ld else ifeq ($(PLATFORM), armv7a) LINKFLAG=-Xlinker -marmelf @@ -82,13 +82,14 @@ COMP_DEP_OBJS=$(foreach D,$(COMP_IFDEPS_CLEAN),$(INTERDIR)/$(D)/cosrt_c_stub.o) # NOTE: we're currently ignoring the *variants* library requirements, # which will break if an interface's code requires a library + comp_header: - $(info | Composing $(COMP_INTERFACE).$(COMP_NAME) for variable $(COMP_VARNAME) by linking with:) + $(info | Composing component in src/components/$(COMP_PATH) for variable $(COMP_VARNAME) by linking with:) $(info | Exported interfaces: $(COMP_INTERFACES_CLEAN)) $(info | Interface dependencies: $(COMP_IFDEPS_CLEAN)) $(info | Libraries: $(DEPENDENCY_LIBS) $(DEPENDENCY_LIBOBJS)) -.PHONY: component +.PHONY: component comp_header component: clean comp_header $(COMPOBJ) $(if $(COMP_INITARGS_FILE), $(CC) $(INCLUDE) $(CFLAGS) -c -o $(COMP_INITARGS_FILE:%.c=%.o) $(COMP_INITARGS_FILE)) $(if $(COMP_TAR_FILE), cp $(COMP_TAR_FILE) $(TAR_SYMBOL_NAME)) @@ -97,6 +98,8 @@ component: clean comp_header $(COMPOBJ) $(MUSLCC) $(COMPNAME).linked_libs_ifs.o $(MUSLCFLAGS) $(LINKFLAG) -o $(COMPNAME).linked_musl.o $(LD) $(LDFLAGS) -Ttext=$(COMP_BASEADDR) -T $(COMP_LD_SCRIPT) -o $(COMP_OUTPUT) $(COMPNAME).linked_musl.o +component_external: + %.o:%.c $(info | [CC] $<: Compiling) @$(CC) $(INCLUDE) $(CFLAGS) -o $@ -c $< @@ -112,7 +115,7 @@ component: clean comp_header $(COMPOBJ) # see the make manual: create the .d dependencies from include # statements. # redirect error output to /dev/null so that it will not display -# errors when cleaning, this does not affect gcc's error output when +# errors when cleaning, this does not affect gcc's error output when # building the system because that is in a different target path %.d:%.c @set +e; rm -f $@; \ diff --git a/src/components/implementation/no_interface/burn/Makefile b/src/components/implementation/no_interface/burn/Makefile new file mode 100644 index 000000000..babc8cf85 --- /dev/null +++ b/src/components/implementation/no_interface/burn/Makefile @@ -0,0 +1,18 @@ +# Required variables used to drive the compilation process. It is OK +# for many of these to be empty. +# +# The set of interfaces that this component exports for use by other +# components. This is a list of the interface names. +INTERFACE_EXPORTS = +# The interfaces this component is dependent on for compilation (this +# is a list of directory names in interface/) +INTERFACE_DEPENDENCIES = +# The library dependencies this component is reliant on for +# compilation/linking (this is a list of directory names in lib/) +LIBRARY_DEPENDENCIES = component +# Note: Both the interface and library dependencies should be +# *minimal*. That is to say that removing a dependency should cause +# the build to fail. The build system does not validate this +# minimality; that's on you! + +include Makefile.subsubdir diff --git a/src/components/implementation/no_interface/burn/burn.c b/src/components/implementation/no_interface/burn/burn.c new file mode 100644 index 000000000..55c22c06f --- /dev/null +++ b/src/components/implementation/no_interface/burn/burn.c @@ -0,0 +1,11 @@ +#include + +void +cos_init(void) +{ + /* Burn all of the cycles! CPUs are glorified space heaters, right? */ + printc("CPU cycle burn: component %ld, thread id %ld, core %d.\n", + cos_compid(), cos_thdid(), cos_coreid()); + + while (1) ; +} diff --git a/src/components/implementation/no_interface/burn/doc.md b/src/components/implementation/no_interface/burn/doc.md new file mode 100644 index 000000000..266be2a6a --- /dev/null +++ b/src/components/implementation/no_interface/burn/doc.md @@ -0,0 +1,12 @@ +## CPU Burn Component + +A simple component that burns CPU cycles, nothing else. +Can be used in test scenarios as background, CPU-intensive computations. + +### Description + +Simple infinite loop, nothing more. + +### Usage and Assumptions + +Only depends on the `init` interface, so it can be used anywhere from on top of the constructor, all the way to as a component under the scheduler. diff --git a/src/components/repos/.gitignore b/src/components/repos/.gitignore new file mode 100644 index 000000000..7c9d611b5 --- /dev/null +++ b/src/components/repos/.gitignore @@ -0,0 +1,3 @@ +* +!.gitignore +!README.md diff --git a/src/components/repos/README.md b/src/components/repos/README.md new file mode 100644 index 000000000..f4262cbff --- /dev/null +++ b/src/components/repos/README.md @@ -0,0 +1,24 @@ +# External Repositories + +Composite current enables components to be defined externally from the main repo, thus providing increased autonomy in development. +Each repository in this directory is pulled from github, and can contain a set of components and composition scripts. +These repositories can be thought of as part of the Composite ecosystem, but not as part of the "standard library" of necessary functionalities. + +A number of awkward realities with the current system: + +- Composite does *not* maintain the repos downloaded here in any way. + If the upstream repo is changed, you must manually do a `git pull` in the repo (or remove it so it is re-cloned). + Comparably, if you update the external repo, you should manually commit/push, etc... +- The Composite repo ignores all repos in here. + Never add them. + The proper course of action would be to do a PR to the main Composite repo adding your component instead. + +## TODO: Libraries and Interfaces + +It should be possible to also define libraries and interfaces in external repositories. +This is more complicated as the dependencies for these are + +1. not properly maintained by the composer (libraries aren't properly rebuilt when one of their dependencies changed), and +2. these dependencies are tracked by the build system, so updating them to enable dependencies are repositories requires most structural updates to the build system. + +There are no hard blockers to doing this; we just need more time and will! diff --git a/src/composer/src/build.rs b/src/composer/src/build.rs index 9d3833da0..280460653 100644 --- a/src/composer/src/build.rs +++ b/src/composer/src/build.rs @@ -40,16 +40,29 @@ use tar::Builder; // - COMP_BASEADDR - the base address of .text for the component // - COMP_INITARGS_FILE - the path to the generated initial arguments .c file // - COMP_TAR_FILE - the path to an initargs tarball to compile into the component +// - COMP_PATH - path to the component directory relative to `src/components` +// +// Building external repositories (not in the main composite repo) +// doesn't use COMP_INTERFACE and instead uses the following: +// +// - REPO_URL - The URL to the to the external repo that holds the +// component. +// - REPO_PATH - The path within the repo directory into which the +// external repo should be cloned. // // In the end, this should result in a command line for each component // along these (artificial) lines: // -// `make COMP_INTERFACES="pong/log" COMP_IFDEPS="capmgr/stubs sched/lock" COMP_LIBS="ps heap" COMP_INTERFACE=pong COMP_NAME=pingpong COMP_VARNAME=pongcomp component` +// `make COMP_INTERFACES="pong/log" COMP_IFDEPS="capmgr/stubs+sched/lock" COMP_LIBS="ps heap" COMP_INTERFACE=pong COMP_NAME=pingpong COMP_PATH="implementation/pong/pingpong COMP_VARNAME=pongcomp component` // // ...which should output the executable pong.pingpong.pongcomp in the // build directory which is the "sealed" version of the component that // is ready for loading. +// A key design goal of the composer and Composite build system is +// that the absolute paths are all taken care of by the build system, +// thus this code focuses on simply properly guiding the build system. + // The key within the initargs for the tarball, the path of the // tarball, and the set of paths to the files to include in the // tarball and name of them within the tarball. @@ -251,45 +264,31 @@ fn comp_gen_make_cmd( let ds = deps(&s, id); let exports = exports(&s, id); - let (_, if_exp) = exports + let if_exp = exports .iter() - .fold((true, String::from("")), |(first, accum), e| { - let mut ifpath = accum.clone(); - if !first { - ifpath.push_str("+"); - } - ifpath.push_str(&e.interface.clone()); - ifpath.push_str("/"); - ifpath.push_str(&e.variant.clone()); - (false, ifpath) - }); - let (_, if_deps) = ds + .map(|e| format!("{}/{}", e.interface, e.variant)) + .collect::>() + .join("+"); + let if_deps = ds .iter() - .fold((true, String::from("")), |(first, accum), d| { - let mut ifpath = accum.clone(); - if !first { - ifpath.push_str("+"); - } - ifpath.push_str(&d.interface.clone()); - ifpath.push_str("/"); - ifpath.push_str(&d.variant.clone()); - (false, ifpath) - }); + .map(|d| format!("{}/{}", d.interface, d.variant)) + .collect::>() + .join("+"); let mut optional_cmds = String::from(""); optional_cmds.push_str(&format!("COMP_INITARGS_FILE={} ", args_file)); if let Some(s) = tar_file { optional_cmds.push_str(&format!("COMP_TAR_FILE={} ", s)); } - let decomp: Vec<&str> = c.source.split(".").collect(); - assert!(decomp.len() == 2); + // unwrap as we've already validated the name. let compid = s.get_named().rmap().get(&c.name).unwrap(); let baseaddr = s.get_address_assignments().component_baseaddr(compid); + let comp_location = s.get_dir_location().dir_location(&c.name); let cmd = format!( - r#"make -C src COMP_INTERFACES="{}" COMP_IFDEPS="{}" COMP_LIBDEPS="" COMP_INTERFACE={} COMP_NAME={} COMP_VARNAME={} COMP_OUTPUT={} COMP_BASEADDR={:#X} {} component"#, - if_exp, if_deps, &decomp[0], &decomp[1], &c.name, output_name, baseaddr, &optional_cmds + r#"make -C src COMP_INTERFACES="{}" COMP_IFDEPS="{}" COMP_LIBDEPS="" COMP_VARNAME={} COMP_OUTPUT={} COMP_PATH={} COMP_BASEADDR={:#X} {} component"#, + if_exp, if_deps, &c.name, output_name, comp_location, baseaddr, &optional_cmds ); cmd @@ -302,6 +301,13 @@ fn kern_gen_make_cmd(input_constructor: &String, kern_output: &String, _s: &Syst ) } +fn repo_download_gen_make_cmd(repo_url: &String, repo_path: &String) -> String { + format!( + r#"make -C src REPO_URL={} REPO_PATH={} repo_external"#, + repo_url, repo_path + ) +} + pub struct DefaultBuilder { builddir: String, } @@ -340,7 +346,7 @@ impl BuildState for DefaultBuilder { fn comp_dir_path(&self, c: &ComponentId, state: &SystemState) -> Result { let name = state.get_named().ids().get(c).unwrap(); - Ok(self.file_path(&format!("{}.{}", name.scope_name, name.var_name))?) + Ok(self.file_path(&format!("{}", name))?) } fn comp_file_path( @@ -357,7 +363,7 @@ impl BuildState for DefaultBuilder { fn comp_obj_file(&self, c: &ComponentId, s: &SystemState) -> String { let comp = component(&s, &c); - format!("{}.{}", &comp.source, &comp.name) + format!("{}", &comp) } fn comp_obj_path(&self, c: &ComponentId, s: &SystemState) -> Result { @@ -461,4 +467,45 @@ impl BuildState for DefaultBuilder { Ok(()) } + + // Take the repo specification from the specification, and the + // path in `src/components/` for the repo. + fn repo_download( + &self, + repo_url: &String, + repo_path: &String, + _s: &SystemState, + ) -> Result<(), String> { + let cmd = repo_download_gen_make_cmd(&repo_url, &repo_path); + println!("Executing `git clone {}`, which may ask for your password...", repo_url); + let (_out, err) = exec_pipeline(vec![cmd.clone()]); + + if err.len() > 0 { + let lines = err.lines(); + let correct_str = "Cloning into "; + let correct_str_len = correct_str.len(); + + let errs: String = lines + .filter_map(|l| { + if l.chars().take(correct_str_len).collect::() != correct_str { + None + } else { + Some(l) + } + }) + .collect(); + + return Err( + format!("Error creating external repository \"{}\":\n", repo_url) + + &errs + + "\nCommon causes of this error include:\n" + + "1. The specified repository doesn't exist (i.e. incorrect URL).\n" + + "2. `git` isn't installed.\n" + + "3. `ssh` isn't installed (we access the repo using git's ssh support).\n" + + "4. `ssh` asked for user input because the known hosts aren't set up.\n", + ); + } + + Ok(()) + } } diff --git a/src/composer/src/cossystem.rs b/src/composer/src/cossystem.rs index 2153754ac..82e2cfe83 100644 --- a/src/composer/src/cossystem.rs +++ b/src/composer/src/cossystem.rs @@ -32,6 +32,7 @@ pub struct InterfaceVariant { pub struct TomlComponent { name: String, img: String, + repo: Option, baseaddr: Option, deps: Option>, params: Option>, @@ -509,6 +510,7 @@ impl Transition for SystemSpec { constructor: ComponentName::new(&c.constructor, &String::from("global")), scheduler: sched_name, source: c.img.clone(), + repo: c.repo.clone(), base_vaddr: c .baseaddr .as_ref() diff --git a/src/composer/src/dir_location.rs b/src/composer/src/dir_location.rs new file mode 100644 index 000000000..924c9bb61 --- /dev/null +++ b/src/composer/src/dir_location.rs @@ -0,0 +1,81 @@ +use passes::{BuildState, ComponentName, DirLocationPass, SystemState, Transition}; +use std::collections::HashMap; + +pub struct Repos { + dirs: HashMap, +} + +impl DirLocationPass for Repos { + fn dir_location(&self, c: &ComponentName) -> &String { + // unwrap valid as a location is added for each component + self.dirs.get(&c).unwrap() + } +} + +impl Transition for Repos { + fn transition(s: &SystemState, b: &mut dyn BuildState) -> Result, String> { + let spec = s.get_spec(); + let mut repos: Vec<&String> = spec + .names() + .iter() + .filter_map(|cn| spec.component_named(cn).repo.as_ref()) + .collect(); + repos.sort(); + repos.dedup(); + let comp_locations = spec.names().iter().map(|c| { + ( + c.clone(), + &spec.component_named(&c).source, + &spec.component_named(&c).repo, + ) + }); + + fn repo_dir_create(r: &String) -> String { + r.replace("/", "-").replace(":", "-") + } + + // Download any repos that aren't already present. + for r in repos { + // Lets parse the repo into the URL to pass to `git`, and into + // the path in the source repo. + let repo_split: Vec<&str> = r.split(':').collect(); + if repo_split.len() != 2 { + return Err(format!( + r#"Repo specification "{}" must have the form : (e.g. "github.com:gparmer/hello_world").\n"#, + r + )); + } + if repo_split[0] != "github" { + return Err(format!( + r#"Repo specification provider "{}" is not supported (supported: "github").\n"#, + repo_split[0] + )); + } + + let repo_url = format!("git@github.com:{}", repo_split[1]); + + b.repo_download(&repo_url, &repo_dir_create(&r), &s)?; + } + + // Add the paths of the components (in external repos, and in + // the standard source) to the DirLocationpass-queryable + // hashmap. + let mut dirs = HashMap::new(); + for (name, source, repo) in comp_locations { + let loc = source + .as_str() + .split(&['.', '/'][..]) + .collect::>() + .join("/"); + dirs.insert( + name, + match repo { + Some(r) => format!("repos/{}/{}", repo_dir_create(&r.clone()), &loc), + None => String::from("implementation/") + &loc, + }, + ); + } + + Ok(Box::new(Repos { dirs })) + } +} diff --git a/src/composer/src/main.rs b/src/composer/src/main.rs index bfb27751d..25b5b8a7f 100644 --- a/src/composer/src/main.rs +++ b/src/composer/src/main.rs @@ -18,6 +18,7 @@ mod syshelpers; mod tot_order; mod properties; mod address_assignment; +mod dir_location; use build::DefaultBuilder; use compobject::{Constructor, ElfObject}; @@ -27,9 +28,10 @@ use invocations::Invocations; use passes::{BuildState, ComponentId, SystemState, Transition, TransitionIter}; use resources::ResAssignPass; use properties::CompProperties; -use std::env; use tot_order::CompTotOrd; use address_assignment::AddressAssignmentx86_64; +use dir_location::Repos; +use std::env; pub fn exec() -> Result<(), String> { let mut args = env::args(); @@ -51,6 +53,7 @@ pub fn exec() -> Result<(), String> { sys.add_parsed(SystemSpec::transition(&sys, &mut build)?); sys.add_named(CompTotOrd::transition(&sys, &mut build)?); + sys.add_dir_location(Repos::transition(&sys, &mut build)?); sys.add_address_assign(AddressAssignmentx86_64::transition(&sys, &mut build)?); sys.add_properties(CompProperties::transition(&sys, &mut build)?); sys.add_restbls(ResAssignPass::transition(&sys, &mut build)?); diff --git a/src/composer/src/passes.rs b/src/composer/src/passes.rs index 352c96a7c..cf64b3092 100644 --- a/src/composer/src/passes.rs +++ b/src/composer/src/passes.rs @@ -21,6 +21,7 @@ pub struct SystemState { parse: Option>, named: Option>, + dir_location: Option>, address_assignment: Option>, properties: Option>, restbls: Option>, @@ -36,6 +37,7 @@ impl SystemState { spec, parse: None, named: None, + dir_location: None, address_assignment: None, properties: None, restbls: None, @@ -54,6 +56,10 @@ impl SystemState { self.named = Some(n); } + pub fn add_dir_location(&mut self, l: Box) { + self.dir_location = Some(l); + } + pub fn add_address_assign(&mut self, a: Box) { self.address_assignment = Some(a); } @@ -94,6 +100,10 @@ impl SystemState { &**(self.named.as_ref().unwrap()) } + pub fn get_dir_location(&self) -> &dyn DirLocationPass { + &**(self.dir_location.as_ref().unwrap()) + } + pub fn get_address_assignments(&self) -> &dyn AddressAssignmentPass { &**(self.address_assignment.as_ref().unwrap()) } @@ -143,6 +153,7 @@ pub trait BuildState { fn comp_build(&self, c: &ComponentId, state: &SystemState) -> Result; // build the component, and return the path to the resulting object fn constructor_build(&self, c: &ComponentId, state: &SystemState) -> Result; // build a constructor, including all components it is responsible for booting fn kernel_build(&self, kern_output: &String, constructor_input: &String, s: &SystemState) -> Result<(), String>; // build the final kernel image + fn repo_download(&self, github_repo: &String, dir_location: &String, s: &SystemState) -> Result<(), String>; } // The following describes the means of transitioning the system @@ -209,12 +220,19 @@ pub struct Component { pub constructor: ComponentName, // the constructor that loads this component pub scheduler: ComponentName, // our scheduler (that creates or initial thread) - pub source: String, // Where is the component source located? + pub source: String, // Where is the component source located in repo? + pub repo: Option, // ...and should we use an external repo? pub base_vaddr: String, // The lowest virtual address for the component -- could be hex, so not a VAddr pub params: Vec, // initialization parameters pub fsimg: Option, } +impl fmt::Display for Component { + fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { + write!(f, "{}.{}", self.source, self.name) + } +} + // Input/frontend pass taking the specification, and outputing the // first intermediate representation. Expected to populate the // Component structure. @@ -251,6 +269,14 @@ pub trait SpecificationPass { fn address_spaces(&self) -> &AddrSpaces; } +// Find and initialize the location in the FS of a component. This is +// in the main Composite repo by default (in +// `components/implementation/*`), and in `components/repos/` otherwise. +pub trait DirLocationPass { + // return the path to the directory to start building the component. + fn dir_location(&self, &ComponentName) -> &String; +} + // Integer namespacing pass. Convert the component variable names to // component ids, and create a total order for the components based on // their stated dependencies (lower ids are more trusted (more @@ -326,7 +352,7 @@ pub trait AddressAssignmentPass { // Compute the resource table, and resource allocations for each // component. pub trait ResPass { - fn args(&self, &ComponentId) -> &Vec; + fn args(&self, _: &ComponentId) -> &Vec; } // The initparam, objects, and synchronous invocation passes are all diff --git a/src/composer/src/syshelpers.rs b/src/composer/src/syshelpers.rs index c660dcfcc..f8ebae726 100644 --- a/src/composer/src/syshelpers.rs +++ b/src/composer/src/syshelpers.rs @@ -62,9 +62,5 @@ pub fn reset_dir(dirname: &String) -> Result<(), String> { } pub fn dir_exists(dirname: &String) -> bool { - if let Err(_) = fs::read_dir(&dirname) { - false - } else { - true - } + fs::read_dir(&dirname).is_ok() }