# Config options ENABLE_LTO ?= n ENABLE_STRIP ?= n # Binaries RUSTC ?= rustc RM := rm # Flags RUSTCFLAGS := --opt-level=3 RMFLAGS := # Handle config setup ifeq ($(ENABLE_LTO),y) RUSTCBINFLAGS := $(RUSTCFLAGS) -Z lto else RUSTCBINFLAGS := $(RUSTCFLAGS) endif ifneq ($(ENABLE_STRIP),y) ENABLE_STRIP := endif # Install directories PREFIX ?= /usr/local BINDIR ?= /bin SRC_DIR=$(shell pwd) # Possible programs PROGS := \ base64 \ basename \ cat \ cksum \ comm \ cp \ cut \ dirname \ echo \ env \ du \ factor \ false \ fmt \ fold \ link \ hashsum \ mkdir \ nl \ paste \ printenv \ pwd \ realpath \ relpath \ rm \ rmdir \ sleep \ split \ seq \ shuf \ sum \ sync \ tac \ tee \ test \ touch \ tr \ true \ truncate \ tsort \ unlink \ uniq \ wc \ yes \ head \ tail \ whoami UNIX_PROGS := \ chroot \ groups \ hostid \ hostname \ id \ kill \ logname \ mkfifo \ nohup \ tty \ uname \ uptime \ users ifneq ($(OS),Windows_NT) PROGS := $(PROGS) $(UNIX_PROGS) endif ALIASES := \ hashsum:md5sum \ hashsum:sha1sum \ hashsum:sha224sum \ hashsum:sha256sum \ hashsum:sha384sum \ hashsum:sha512sum BUILD ?= $(PROGS) # Output names EXES := \ $(sort $(filter $(BUILD),$(filter-out $(DONT_BUILD),$(PROGS)))) CRATE_RLIBS := INSTALL ?= $(EXES) INSTALLEES := \ $(filter $(INSTALL),$(filter-out $(DONT_INSTALL),$(EXES) uutils)) # Programs with usable tests TEST_PROGS := \ cat \ mkdir \ nl \ seq \ tr \ truncate \ TEST ?= $(TEST_PROGS) TESTS := \ $(filter $(TEST),$(filter-out $(DONT_TEST),$(filter $(BUILD),$(filter-out $(DONT_BUILD),$(TEST_PROGS))))) # Setup for building crates define BUILD_SETUP X := $(shell $(RUSTC) --print-file-name --crate-type rlib $(1)/$(1).rs) $(1)_RLIB := $$(X) CRATE_RLIBS += $$(X) endef $(foreach crate,$(EXES),$(eval $(call BUILD_SETUP,$(crate)))) # Utils stuff EXES_PATHS := $(addprefix build/,$(EXES)) RLIB_PATHS := $(addprefix build/,$(CRATE_RLIBS)) command = sh -c '$(1)' # Main exe build rule define EXE_BUILD build/gen/$(1).rs: build/mkmain build/mkmain $(1) build/gen/$(1).rs build/$(1): build/gen/$(1).rs build/$($(1)_RLIB) | build deps $(RUSTC) $(RUSTCBINFLAGS) -L build/ -o build/$(1) build/gen/$(1).rs $(if $(ENABLE_STRIP),strip build/$(1),) endef define CRATE_BUILD -include build/$(1).d build/$($(1)_RLIB): $(1)/$(1).rs | build deps $(RUSTC) $(RUSTCFLAGS) -L build/ --crate-type rlib --dep-info build/$(1).d $(1)/$(1).rs --out-dir build endef # Aliases build rule ALIAS_SOURCE = $(firstword $(subst :, ,$(1))) ALIAS_TARGET = $(word 2,$(subst :, ,$(1))) define MAKE_ALIAS ifneq ($(ALIAS_TARGET,$(1)),) all: build/$(call ALIAS_TARGET,$(1)) build/$(call ALIAS_TARGET,$(1)): build/$(call ALIAS_SOURCE,$(1)) $(call command,install build/$(call ALIAS_SOURCE,$(1)) build/$(call ALIAS_TARGET,$(1))) endif endef # Test exe built rules define TEST_BUILD test_$(1): tmp/$(1)_test build/$(1) $(call command,tmp/$(1)_test) tmp/$(1)_test: $(1)/test.rs $(call command,$(RUSTC) $(RUSTCFLAGS) --test -o tmp/$(1)_test $(1)/test.rs) endef # Main rules all: $(EXES_PATHS) build/uutils # Creating necessary rules for each targets $(foreach crate,$(EXES),$(eval $(call CRATE_BUILD,$(crate)))) $(foreach exe,$(EXES),$(eval $(call EXE_BUILD,$(exe)))) $(foreach alias,$(ALIASES),$(eval $(call MAKE_ALIAS,$(alias)))) $(foreach test,$(TESTS),$(eval $(call TEST_BUILD,$(test)))) -include build/uutils.d build/uutils: uutils/uutils.rs build/mkuutils $(RLIB_PATHS) build/mkuutils build/gen/uutils.rs $(BUILD) $(RUSTC) $(RUSTCBINFLAGS) -L build/ --dep-info $@.d build/gen/uutils.rs -o $@ $(if $(ENABLE_STRIP),strip build/uutils) # Dependencies -include build/rust-crypto.d build/.rust-crypto: | build $(RUSTC) $(RUSTCFLAGS) --crate-type rlib --dep-info build/rust-crypto.d deps/rust-crypto/src/rust-crypto/lib.rs --out-dir build/ @touch $@ build/mkmain: mkmain.rs | build $(RUSTC) $(RUSTCFLAGS) -L build mkmain.rs -o $@ build/mkuutils: mkuutils.rs | build $(RUSTC) $(RUSTCFLAGS) -L build mkuutils.rs -o $@ cksum/crc_table.rs: cksum/gen_table.rs cd cksum && $(RUSTC) $(RUSTCFLAGS) gen_table.rs && ./gen_table && $(RM) gen_table deps: build/.rust-crypto cksum/crc_table.rs crates: echo $(EXES) test: tmp $(addprefix test_,$(TESTS)) $(RM) -rf tmp clean: $(RM) -rf build tmp build: git submodule update --init mkdir -p build/gen tmp: mkdir tmp install: $(addprefix build/,$(INSTALLEES)) mkdir -p $(DESTDIR)$(PREFIX)$(BINDIR) for prog in $(INSTALLEES); do \ install build/$$prog $(DESTDIR)$(PREFIX)$(BINDIR)/$(PROG_PREFIX)$$prog; \ done # TODO: figure out if there is way for prefixes to work with the symlinks install-multicall: build/uutils mkdir -p $(DESTDIR)$(PREFIX)$(BINDIR) install build/uutils $(DESTDIR)$(PREFIX)$(BINDIR)/$(PROG_PREFIX)uutils cd $(DESTDIR)$(PREFIX)$(BINDIR) for prog in $(INSTALLEES); do \ ln -s $(PROG_PREFIX)uutils $$prog; \ done uninstall: rm -f $(addprefix $(DESTDIR)$(PREFIX)$(BINDIR)/$(PROG_PREFIX),$(PROGS)) uninstall-multicall: rm -f $(addprefix $(DESTDIR)$(PREFIX)$(BINDIR)/,$(PROGS) $(PROG_PREFIX)uutils) # Test under the busybox testsuite build/busybox: build/uutils rm -f build/busybox ln -s $(SRC_DIR)/build/uutils build/busybox # This is a busybox-specific config file their test suite wants to parse. # For now it's blank. build/.config: build/uutils touch $@ ifeq ($(BUSYBOX_SRC),) busytest: @echo @echo "To run \`busytest\` set BUSYBOX_SRC to the directory of the compiled busybox source code." @echo "Optionally set RUNTEST_ARGS to arguments to pass to the busybox \`runtest\` program." @echo @false else busytest: build/busybox build/.config (cd $(BUSYBOX_SRC)/testsuite && bindir=$(SRC_DIR)/build ./runtest $(RUNTEST_ARGS)) endif .PHONY: all deps test clean busytest install uninstall