⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 build.cxx

📁 基于ecos的redboot
💻 CXX
📖 第 1 页 / 共 2 页
字号:
	std::string command_prefix = get_flags (config, NULL, "COMMAND_PREFIX");
	if (! command_prefix.empty ()) { // if there is a command prefix
		command_prefix += '-'; // add a trailing hyphen
	}

	// generate the prefix for archived objects
	unsigned int final_separator = 0; // the index of the last directory separator
	std::string object_prefix = info.directory; // start with the loadable directory
	for (count = 0; count < object_prefix.size (); count++) { // for each char
		if ('/' == object_prefix [count]) { // if the char is a directory separator
			object_prefix [count] = '_'; // replace the char with an underscore
			final_separator = count;
		}
	}
	object_prefix.resize (final_separator); // remove the version directory
	
	// open the makefile
	FILE * stream = fopen (filename.c_str (), "wt");
	if (stream == NULL) // if the file could not be opened
		return false;

	// generate the header
	fprintf (stream, makefile_header.c_str ());

	// generate the global variables
	fprintf (stream, "export REPOSITORY := %s\n", cygpath (config->get_database ()->get_component_repository ()).c_str ());
	fprintf (stream, "export PREFIX := %s\n", cygpath (install_tree).c_str ());
	fprintf (stream, "export COMMAND_PREFIX := %s\n", command_prefix.c_str ());
	fprintf (stream, "export CC := $(COMMAND_PREFIX)gcc\n");
	fprintf (stream, "export OBJCOPY := $(COMMAND_PREFIX)objcopy\n");
#ifdef _WIN32
    fprintf (stream, "export HOST := CYGWIN\n");
#else
    fprintf (stream, "export HOST := UNIX\n");
#endif
	fprintf (stream, "export AR := $(COMMAND_PREFIX)ar\n\n");

	// generate the package variables
	fprintf (stream, "PACKAGE := %s\n", info.directory.c_str ());
	fprintf (stream, "OBJECT_PREFIX := %s\n", object_prefix.c_str ());
	fprintf (stream, "CFLAGS := %s\n", get_flags (config, &info, "CFLAGS").c_str ());
	fprintf (stream, "LDFLAGS := %s\n", get_flags (config, &info, "LDFLAGS").c_str ());
	fprintf (stream, "VPATH := $(REPOSITORY)/$(PACKAGE)\n");
	fprintf (stream, "INCLUDE_PATH := $(INCLUDE_PATH) -I$(PREFIX)/include $(foreach dir,$(VPATH),-I$(dir) -I$(dir)/src -I$(dir)/tests) -I.\n");
	fprintf (stream, "MLT := $(wildcard $(REPOSITORY)/$(PACKAGE)/include/pkgconf/mlt*.ldi $(REPOSITORY)/$(PACKAGE)/include/pkgconf/mlt*.h)\n");
	fprintf (stream, "TESTS := %s\n\n", get_tests (config, info).c_str ());

	// create a vector of libraries
	std::vector <std::string> library_vector;
	for (count = 0; count < info.compiles.size (); count++) { // for each compilable file
		const std::string & library = info.compiles [count].library;
		if (library_vector.end () == std::find (library_vector.begin (), library_vector.end (), library)) { // if a new library
			library_vector.push_back (library); // add the library to the vector
		}
	}
	for (count = 0; count < info.objects.size (); count++) { // for each object file
		const std::string & library = info.objects [count].library;
		if (library_vector.end () == std::find (library_vector.begin (), library_vector.end (), library)) { // if a new library
			library_vector.push_back (library); // add the library to the vector
		}
	}
	for (count = 0; count < info.make_objects.size (); count++) { // for each make object
		const std::string & library = info.make_objects [count].library;
		if (library_vector.end () == std::find (library_vector.begin (), library_vector.end (), library)) { // if a new library
			library_vector.push_back (library); // add the library to the vector
		}
	}

	// generate the default rule
	fprintf (stream, "build: headers");
	for (library = 0; library < library_vector.size (); library++) { // for each library
		fprintf (stream, " %s.stamp", library_vector [library].c_str ());
	}
	fprintf (stream, "\n\n");

	// generate library rules
	for (library = 0; library < library_vector.size (); library++) { // for each library
		fprintf (stream, "LIBRARY := %s\n", library_vector [library].c_str ());
		fprintf (stream, "COMPILE :=");
		for (count = 0; count < info.compiles.size (); count++) { // for each compilable file
			if (library_vector [library] == info.compiles [count].library) { // if the file and library are related
				fprintf (stream, " %s", info.compiles [count].source.c_str ());
			}
		}
		for (count = 0; count < info.objects.size (); count++) { // for each object file
			if (library_vector [library] == info.objects [count].library) { // if the file and library are related
				fprintf (stream, " %s", info.objects [count].object.c_str ());
			}
		}
		for (count = 0; count < info.make_objects.size (); count++) { // for each make object
			if (library_vector [library] == info.make_objects [count].library) { // if the object and library are related
				fprintf (stream, " %s", info.make_objects [count].object.c_str ());
			}
		}

		fprintf (stream, "\nOBJECTS := $(COMPILE:.cxx=.o.d)\n");
		fprintf (stream, "OBJECTS := $(OBJECTS:.c=.o.d)\n");
		fprintf (stream, "OBJECTS := $(OBJECTS:.S=.o.d)\n\n");
		fprintf (stream, "$(LIBRARY).stamp: $(OBJECTS)\n");
		fprintf (stream, "\t$(AR) rcs $(PREFIX)/lib/$(@:.stamp=) $(foreach obj,$?,$(dir $(obj))$(OBJECT_PREFIX)_$(notdir $(obj:.o.d=.o)))\n");
		fprintf (stream, "\t@cat $^ > $(@:.stamp=.deps)\n");
		fprintf (stream, "\t@touch $@\n\n");
	}	

	// generate make objects rules
	for (count = 0; count < info.make_objects.size (); count++) { // for each make object
		fprintf (stream, "%s: %s\n%s\n", info.make_objects [count].object.c_str (), info.make_objects [count].deps.c_str (), tab_indent (info.make_objects [count].rules).c_str ());
	}

	// generate makes rules
	for (count = 0; count < info.makes.size (); count++) { // for each make
		fprintf (stream, "%s: $(wildcard %s)\n%s\n", resolve_tokens (info.makes [count].target).c_str (), resolve_tokens (info.makes [count].deps).c_str (), tab_indent (info.makes [count].rules).c_str ());
	}

	// generate header rules
	fprintf (stream, "headers: mlt_headers");
	for (count = 0; count < info.headers.size (); count++) { // for each header
		fprintf (stream, " $(PREFIX)/include/%s", info.headers [count].destination.c_str ());
	}
	fprintf (stream, "\n\n");
	for (count = 0; count < info.headers.size (); count++) { // for each header
		fprintf (stream, "$(PREFIX)/include/%s: $(REPOSITORY)/$(PACKAGE)/%s\n", info.headers [count].destination.c_str (), info.headers [count].source.c_str ());
#if defined(_WIN32) && (ECOS_USE_CYGDRIVE == 1)
        fprintf (stream, "ifeq ($(HOST),CYGWIN)\n");
	    fprintf (stream, "\t@mkdir -p `cygpath -w \"$(dir $@)\" | sed \"s/\\\\\\\\\\/\\\\//g\"`\n");
        fprintf (stream, "else\n");
	    fprintf (stream, "\t@mkdir -p $(dir $@)\n");
        fprintf (stream, "endif\n");
#else
        // This prevents older versions of mkdir failing
        fprintf (stream, "\t@mkdir -p $(dir $@)\n");
#endif

		fprintf (stream, "\t@cp $< $@\n");
		fprintf (stream, "\t@chmod u+w $@\n\n");
	}

	// include default rules
	fprintf (stream, "include $(REPOSITORY)/pkgconf/rules.mak\n\n");

	// close the makefile
	return (0 == fclose (stream));
}

// a structure and operator for sorting custom make rules by priority
struct info_make {
	std::vector <CdlBuildInfo_Loadable>::const_iterator loadable;
	unsigned int make;
};
bool operator < (info_make op1, info_make op2) {
	return op1.loadable->makes [op1.make].priority < op2.loadable->makes [op2.make].priority;
}

// create the top-level makefile
bool generate_toplevel_makefile (const CdlConfiguration config, const std::string install_tree, const std::string filename) {
	unsigned int loadable;
	unsigned int make;

	// obtain the command prefix
	std::string command_prefix = get_flags (config, NULL, "COMMAND_PREFIX");
	if (! command_prefix.empty ()) { // if there is a command prefix
		command_prefix += '-'; // add a trailing hyphen
	}

	// obtain build information from the specified configuration
	CdlBuildInfo build_info;
	config->get_build_info (build_info);
	std::vector <CdlBuildInfo_Loadable> info_vector = build_info.entries;

	// create a vector of makes and sort them by priority
	std::vector <info_make> info_make_vector;
	for (std::vector <CdlBuildInfo_Loadable>::iterator info = info_vector.begin (); info != info_vector.end (); info++) { // for each buildable loaded package
		for (unsigned int count = 0; count < info->makes.size (); count++) { // for each make
			info_make make_info = {info, count};
			info_make_vector.push_back (make_info);
		}
	}
	std::sort (info_make_vector.begin (), info_make_vector.end ());

	// open the makefile
	FILE * stream = fopen (filename.c_str (), "wt");
	if (stream == NULL) // if the file could not be opened
		return false;

	// generate the header
	fprintf (stream, makefile_header.c_str ());

	// generate the variables
	fprintf (stream, "export REPOSITORY := %s\n", cygpath (config->get_database ()->get_component_repository ()).c_str ());
#ifdef _WIN32
    fprintf (stream, "export HOST := CYGWIN\n");
#else
    fprintf (stream, "export HOST := UNIX\n");
#endif
	fprintf (stream, "export PREFIX := %s\n", cygpath (install_tree).c_str ());
	fprintf (stream, "export COMMAND_PREFIX := %s\n", command_prefix.c_str ());
	fprintf (stream, "export CC := $(COMMAND_PREFIX)gcc\n");
	fprintf (stream, "export OBJCOPY := $(COMMAND_PREFIX)objcopy\n");
	fprintf (stream, "export AR := $(COMMAND_PREFIX)ar\n\n");

    // generate the makefile contents
	fprintf (stream, ".PHONY: default build clean tests headers\n\n");

	fprintf (stream, "build: headers $(PREFIX)/include/pkgconf/ecos.mak\n");
	for (make = 0; make < info_make_vector.size (); make++) { // for each make
		if (info_make_vector [make].loadable->makes [info_make_vector [make].make].priority < 100) { // if priority higher than default complilation
			fprintf (stream, "\t$(MAKE) -r -C %s %s\n", info_make_vector [make].loadable->directory.c_str (), resolve_tokens (info_make_vector [make].loadable->makes [info_make_vector [make].make].target).c_str ());
		}
	}
	for (loadable = 0; loadable < info_vector.size (); loadable++) { // for each buildable loaded package
		const std::string source_path = info_vector [loadable].directory;
		fprintf (stream, "\t$(MAKE) -r -C %s $@\n", source_path.c_str ());
	}
	for (make = 0; make < info_make_vector.size (); make++) { // for each make
		if (info_make_vector [make].loadable->makes [info_make_vector [make].make].priority >= 100) { // if priority lower than or equal to default complilation
			fprintf (stream, "\t$(MAKE) -r -C %s %s\n", info_make_vector [make].loadable->directory.c_str (), resolve_tokens (info_make_vector [make].loadable->makes [info_make_vector [make].make].target).c_str ());
		}
	}
	fprintf (stream, "\t@echo $@ finished\n\n");

	fprintf (stream, "clean:\n");
	for (loadable = 0; loadable < info_vector.size (); loadable++) { // for each buildable loaded package
		const std::string source_path = info_vector [loadable].directory;
		fprintf (stream, "\t$(MAKE) -r -C %s $@\n", source_path.c_str ());
	}
	fprintf (stream, "\t@echo $@ finished\n\n");

	fprintf (stream, "tests: build\n");
	for (loadable = 0; loadable < info_vector.size (); loadable++) { // for each buildable loaded package
		const std::string source_path = info_vector [loadable].directory;
		fprintf (stream, "\t$(MAKE) -r -C %s $@\n", source_path.c_str ());
	}
	fprintf (stream, "\t@echo $@ finished\n\n");

	fprintf (stream, "headers:\n");
	for (loadable = 0; loadable < info_vector.size (); loadable++) { // for each buildable loaded package
		const std::string source_path = info_vector [loadable].directory;
		fprintf (stream, "\t$(MAKE) -r -C %s $@\n", source_path.c_str ());
	}
	fprintf (stream, "\t@echo $@ finished\n\n");

	fprintf (stream, "$(PREFIX)/include/pkgconf/ecos.mak: makefile\n");
	fprintf (stream, "\t@echo 'ECOS_GLOBAL_CFLAGS = %s' > $@\n", get_flags (config, NULL, "CFLAGS").c_str ());
	fprintf (stream, "\t@echo 'ECOS_GLOBAL_LDFLAGS = %s' >> $@\n", get_flags (config, NULL, "LDFLAGS").c_str ());
	fprintf (stream, "\t@echo 'ECOS_COMMAND_PREFIX = $(COMMAND_PREFIX)' >> $@\n\n");

	// close the makefile
	return (0 == fclose (stream));
}

// generates the directory structure for the build and install trees
bool generate_build_tree (const CdlConfiguration config, const std::string build_tree, const std::string install_tree /* = "" */ ) {
/* #ifdef _WIN32 */
#if 1 /* Lomesh */
	// convert backslash directory separators to forward slashes under Win32
	const std::string build_dir = replace_char (build_tree, '\\', '/');
	const std::string install_dir = install_tree.empty () ? build_dir + "/install" : replace_char (install_tree, '\\', '/');
#else
	const std::string build_dir = build_tree;
	const std::string install_dir = install_tree.empty () ? build_dir + "/install" : install_tree;
#endif

	// create build and install directories to ensure they are in writable locations
	if (! create_directory (build_dir))
		return false;
	if (! create_directory (install_dir + "/lib"))
		return false;
	if (! create_directory (install_dir + "/include/pkgconf"))
		return false;

	// obtain build information from the specified configuration
	CdlBuildInfo build_info;
	config->get_build_info (build_info);
	std::vector <CdlBuildInfo_Loadable> info_vector = build_info.entries;

	for (unsigned int loadable = 0; loadable < info_vector.size (); loadable++) { // for each buildable loaded package
		const std::string build_dir_loadable = build_dir + "/" + info_vector [loadable].directory;

		// create loadable directory in build tree
		if (! create_directory (build_dir_loadable))
			return false;

		// generate makefile
		generate_makefile (config, info_vector [loadable], install_dir, build_dir_loadable + "/makefile");
	}

	generate_toplevel_makefile (config, install_dir, build_dir + "/makefile");

	return true;
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -