-#define _CRT_SECURE_NO_DEPRECATE\r
-\r
-#include <windows.h>\r
-#include <stdio.h>\r
-#include <string>\r
-#include "colours.h"\r
-\r
-using namespace std;\r
-void Run();\r
-void Banner();\r
-void WriteCompileModules();\r
-void WriteCompileCommands();\r
-\r
-/* detects if we are running windows xp or higher (5.1) */\r
-bool iswinxp()\r
-{\r
- OSVERSIONINFO vi;\r
- vi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);\r
- GetVersionEx(&vi);\r
- if(vi.dwMajorVersion > 5)\r
- return true;\r
-\r
- if(vi.dwMajorVersion >= 5 && vi.dwMinorVersion >= 1)\r
- return true;\r
- \r
- return false;\r
-}\r
-\r
-int get_int_option(const char * text, int def)\r
-{\r
- static char buffer[500];\r
- int ret;\r
- printf_c("%s\n[\033[1;32m%u\033[0m] -> ", text, def);\r
- fgets(buffer, 500, stdin);\r
- if(sscanf(buffer, "%u", &ret) != 1)\r
- ret = def;\r
-\r
- printf("\n");\r
- return ret;\r
-}\r
-\r
-bool get_bool_option(const char * text, bool def)\r
-{\r
- static char buffer[500];\r
- char ret[100];\r
- printf_c("%s [\033[1;32m%c\033[0m] -> ", text, def ? 'y' : 'n');\r
- fgets(buffer, 500, stdin);\r
- if(sscanf(buffer, "%s", ret) != 1)\r
- strcpy(ret, def ? "y" : "n");\r
-\r
- printf("\n");\r
- return !strncmp(ret, "y", 1);\r
-}\r
-\r
-void get_string_option(const char * text, char * def, char * buf)\r
-{\r
- static char buffer[500];\r
- printf_c("%s\n[\033[1;32m%s\033[0m] -> ", text, def);\r
- fgets(buffer, 500, stdin);\r
- if(sscanf(buffer, "%s", buf) != 1)\r
- strcpy(buf, def);\r
-\r
- printf("\n");\r
-}\r
-\r
-// escapes a string for use in a c++ file\r
-bool escape_string(char * str, size_t size)\r
-{\r
- size_t len = strlen(str);\r
- char * d_str = (char*)malloc(len * 2);\r
- \r
- size_t i = 0;\r
- size_t j = 0;\r
-\r
- for(; i < len; ++i)\r
- {\r
- if(str[i] == '\\')\r
- {\r
- d_str[j++] = '\\';\r
- d_str[j++] = '\\';\r
- }\r
- else\r
- {\r
- d_str[j++] = str[i];\r
- }\r
- }\r
-\r
- d_str[j++] = 0;\r
-\r
- if(j > size)\r
- {\r
- free(d_str);\r
- return false;\r
- }\r
-\r
- strcpy(str, d_str);\r
- free(d_str);\r
- return true;\r
-}\r
-\r
-/* gets the svn revision */\r
-int get_svn_revision(char * buffer, size_t len)\r
-{\r
- /* again.. I am lazy :p cbf to pipe output of svn info to us, so i'll just read the file */\r
- /*\r
- 8\r
-\r
- dir\r
- 7033\r
- */\r
- char buf[1000];\r
- FILE * f = fopen("..\\.svn\\entries", "r");\r
- if(!f) goto bad_rev;\r
- \r
- if(!fgets(buf, 1000, f)) goto bad_rev;\r
- if(!fgets(buf, 1000, f)) goto bad_rev;\r
- if(!fgets(buf, 1000, f)) goto bad_rev;\r
- if(!fgets(buf, 1000, f)) goto bad_rev;\r
- int rev = atoi(buf);\r
- if(rev == 0) goto bad_rev;\r
- sprintf(buffer, "%u", rev);\r
- fclose(f);\r
- return rev;\r
- \r
-bad_rev:\r
- strcpy(buffer, "non-svn");\r
- if(f) fclose(f);\r
- return 0;\r
-}\r
-\r
-int __stdcall WinMain(IN HINSTANCE hInstance, IN HINSTANCE hPrevInstance, IN LPSTR lpCmdLine, IN int nShowCmd )\r
-{\r
- // Skip if configure is already-existant.\r
- FILE * f = fopen("inspircd_config.h", "r");\r
- if(f)\r
- {\r
- fclose(f);\r
- return 0;\r
- }\r
-\r
- AllocConsole();\r
-\r
- // pipe standard handles to this console\r
- freopen("CONIN$", "r", stdin);\r
- freopen("CONOUT$", "w", stdout);\r
- freopen("CONOUT$", "w", stderr);\r
-\r
- Banner();\r
- Run();\r
- WriteCompileCommands();\r
- WriteCompileModules();\r
- FreeConsole();\r
- return 0;\r
-}\r
-\r
-void Banner()\r
-{\r
- printf_c("\nWelcome to the \033[1mInspIRCd\033[0m Configuration program! (\033[1minteractive mode\033[0m)\n"\r
- "\033[1mPackage maintainers: Type ./configure --help for non-interactive help\033[0m\n\n");\r
- printf_c("*** If you are unsure of any of these values, leave it blank for ***\n"\r
- "*** standard settings that will work, and your server will run ***\n"\r
- "*** using them. Please consult your IRC network admin if in doubt. ***\n\n"\r
- "Press \033[1m<RETURN>\033[0m to accept the default for any option, or enter\n"\r
- "a new value. Please note: You will \033[1mHAVE\033[0m to read the docs\n"\r
- "dir, otherwise you won't have a config file!\n\n");\r
-\r
-}\r
-\r
-void Run()\r
-{\r
- int max_fd = 1024;\r
- bool use_iocp = false;\r
- bool support_ip6links = false;\r
- char mod_path[MAX_PATH];\r
- char config_file[MAX_PATH];\r
- char library_dir[MAX_PATH];\r
- char base_path[MAX_PATH];\r
- char bin_dir[MAX_PATH];\r
- char revision_text[MAX_PATH];\r
-\r
- int max_clients = 1024;\r
- int nicklen = 31;\r
- int chanlen = 64;\r
- int modechanges = 20;\r
- int identlen = 12;\r
- int quitlen = 255;\r
- int topiclen = 500;\r
- int kicklen = 255;\r
- int rllen = 128;\r
- int awaylen = 200;\r
- int revision = get_svn_revision(revision_text, MAX_PATH);\r
- char version[514];\r
-\r
- // grab version\r
- FILE * fI = fopen("..\\src\\version.sh", "r");\r
- if(fI)\r
- {\r
- fgets(version, 514, fI);\r
- fgets(version, 514, fI);\r
- char * p2 = version;\r
- while(*p2 != '\"')\r
- ++p2;\r
- ++p2;\r
- strcpy(version, p2);\r
- p2 = version;\r
- while(*p2 != '\"')\r
- ++p2;\r
- *p2 = 0;\r
- fclose(fI);\r
- }\r
- else\r
- strcpy(version, "InspIRCD-Unknown");\r
- printf_c("Your operating system is: \033[1;32mwindows \033[0m\n");\r
- printf_c("InspIRCd revision ID: \033[1;32m%s \033[0m\n\n", revision ? revision_text : "(Non-SVN build)");\r
-\r
- max_fd = get_int_option("What is the maximum file descriptor count you would like to allow?", 1024);\r
-\r
- // detect windows\r
- if(iswinxp())\r
- {\r
- printf_c("You are running Windows XP or above, and IOCP support is most likely available.\n"\r
- "This removes the socket number limitation of select and is much more efficent.\n"\r
- "If you are unsure, answer yes.\n\n");\r
-\r
- use_iocp = get_bool_option("Do you want to use the IOCP implementation?", true);\r
- }\r
-\r
- support_ip6links = get_bool_option("\nYou have chosen to build an \033[1;32mIPV4-only\033[0m server.\nWould you like to enable support for linking to IPV6-enabled InspIRCd servers?\nIf you are using a recent operating system and are unsure, answer yes.\nIf you answer 'no' here, your InspIRCd server will be unable\nto parse IPV6 addresses (e.g. for CIDR bans)", \r
- true);\r
- \r
- printf_c("\033[1mAll paths are relative to the binary directory.\033[0m\n");\r
- get_string_option("In what directory do you wish to install the InspIRCd base?", "..", base_path);\r
- get_string_option("In what directory are the configuration files?", "../conf", config_file);\r
- get_string_option("In what directory are the modules to be compiled to?", "../modules", mod_path);\r
- get_string_option("In what directory is the IRCd binary to be placed?", ".", bin_dir);\r
- get_string_option("In what directory are the IRCd libraries to be placed?", "../lib", library_dir);\r
-\r
- printf_c("The following questions will ask you for various figures relating\n"\r
- "To your IRCd install. Please note that these should usually be left\n"\r
- "as defaults unless you have a real reason to change them. If they\n"\r
- "changed, then the values must be identical on all servers on your\n"\r
- "network, or malfunctions and/or crashes may occur, with the exception\n"\r
- "of the 'maximum number of clients' setting which may be different on\n"\r
- "different servers on the network.\n\n");\r
-\r
- \r
- max_clients = get_int_option("Please enter the maximum number of clients at any one time?", 1024);\r
- nicklen = get_int_option("Please enter the maximum length of nicknames?", 31);\r
- chanlen = get_int_option("Please enter the maximum length of channel names?", 64);\r
- modechanges = get_int_option("Please enter the maximum number of mode changes in one line?", 20);\r
- identlen = get_int_option("Please enter the maximum length of an ident (username)?", 12);\r
- quitlen = get_int_option("Please enter the maximum length of a quit message?", 255);\r
- topiclen = get_int_option("Please enter the maximum length of a channel topic?", 307);\r
- kicklen = get_int_option("Please enter the maximum length of a kick message?", 255);\r
- rllen = get_int_option("Please enter the maximum length of a GECOS (real name)?", 128);\r
- awaylen = get_int_option("Please enter the maximum length of an away message?", 200);\r
-\r
- printf_c("\n\033[1;32mPre-build configuration is complete!\n\n"); sc(TNORMAL);\r
-\r
- // dump all the options back out\r
- printf_c("\033[0mBase install path:\033[1;32m %s\n", base_path);\r
- printf_c("\033[0mConfig path:\033[1;32m %s\n", config_file);\r
- printf_c("\033[0mModule path:\033[1;32m %s\n", mod_path);\r
- printf_c("\033[0mLibrary path:\033[1;32m %s\n", library_dir);\r
- printf_c("\033[0mSocket Engine:\033[1;32m %s\n", use_iocp ? "iocp" : "select");\r
- printf_c("\033[0mMax file descriptors:\033[1;32m %u\n", max_fd);\r
- printf_c("\033[0mMax connections:\033[1;32m %u\n", max_clients);\r
- printf_c("\033[0mMax nickname length:\033[1;32m %u\n", nicklen);\r
- printf_c("\033[0mMax channel length:\033[1;32m %u\n", chanlen);\r
- printf_c("\033[0mMax mode length:\033[1;32m %u\n", modechanges);\r
- printf_c("\033[0mMax ident length:\033[1;32m %u\n", identlen);\r
- printf_c("\033[0mMax quit length:\033[1;32m %u\n", quitlen);\r
- printf_c("\033[0mMax topic length:\033[1;32m %u\n", topiclen);\r
- printf_c("\033[0mMax kick length:\033[1;32m %u\n", kicklen);\r
- printf_c("\033[0mMax name length:\033[1;32m %u\n", rllen);\r
- printf_c("\033[0mMax away length:\033[1;32m %u\n", awaylen);\r
- printf("\n"); sc(TNORMAL);\r
- if(get_bool_option("Are these settings correct?", true) == false)\r
- {\r
- Run();\r
- return;\r
- }\r
- printf("\n");\r
-\r
- // escape the pathes\r
- escape_string(config_file, MAX_PATH);\r
- escape_string(mod_path, MAX_PATH);\r
- escape_string(library_dir, MAX_PATH);\r
-\r
- printf("\nWriting inspircd_config.h...");\r
- FILE * f = fopen("inspircd_config.h", "w");\r
- fprintf(f, "/* Auto generated by configure, do not modify! */\n");\r
- fprintf(f, "#ifndef __CONFIGURATION_AUTO__\n");\r
- fprintf(f, "#define __CONFIGURATION_AUTO__\n\n");\r
- if(use_iocp)\r
- fprintf(f, "#define CONFIG_USE_IOCP 1\n\n");\r
-\r
- fprintf(f, "#define CONFIG_FILE \"%s/inspircd.conf\"\n", config_file);\r
- fprintf(f, "#define MOD_PATH \"%s\"\n", mod_path);\r
- fprintf(f, "#define MAX_DESCRIPTORS %u\n", max_fd);\r
- fprintf(f, "#define MAXCLIENTS %u\n", max_clients);\r
- fprintf(f, "#define MAXCLIENTS_S \"%u\"\n", max_clients);\r
- fprintf(f, "#define SOMAXCONN_S \"128\"\n");\r
- fprintf(f, "#define NICKMAX %u\n", nicklen+1);\r
- fprintf(f, "#define CHANMAX %u\n", chanlen+1);\r
- fprintf(f, "#define MAXMODES %u\n", modechanges);\r
- fprintf(f, "#define IDENTMAX %u\n", identlen);\r
- fprintf(f, "#define MAXQUIT %u\n", quitlen);\r
- fprintf(f, "#define MAXTOPIC %u\n", topiclen);\r
- fprintf(f, "#define MAXKICK %u\n", kicklen);\r
- fprintf(f, "#define MAXGECOS %u\n", rllen);\r
- fprintf(f, "#define MAXAWAY %u\n", awaylen);\r
- fprintf(f, "#define LIBRARYDIR \"%s\"\n", library_dir);\r
- fprintf(f, "#define VERSION \"%s\"\n", version);\r
- fprintf(f, "#define REVISION \"%s\"\n", revision_text);\r
- if(support_ip6links)\r
- fprintf(f, "#define SUPPORT_IP6LINKS 1\n");\r
-\r
- OSVERSIONINFO vi;\r
- vi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);\r
- GetVersionEx(&vi);\r
- fprintf(f, "#define SYSTEM \"Windows %u.%u.%u %s\"\n", vi.dwMajorVersion, vi.dwMinorVersion, vi.dwBuildNumber, vi.szCSDVersion);\r
- fprintf(f, "#define MAXBUF 514\n");\r
-\r
- fprintf(f, "\n#include \"inspircd_win32wrapper.h\"\n\n");\r
- fprintf(f, "#endif\n\n");\r
- fclose(f);\r
-\r
- sc(TGREEN); printf(" done\n"); sc(TNORMAL);\r
- printf("Writing inspircd_se_config.h...");\r
-\r
- f = fopen("inspircd_se_config.h", "w");\r
- fprintf(f, "/* Auto generated by configure, do not modify or commit to svn! */\n");\r
- fprintf(f, "#ifndef __CONFIGURATION_SOCKETENGINE__\n");\r
- fprintf(f, "#define __CONFIGURATION_SOCKETENGINE__\n\n");\r
- fprintf(f, "#include \"socketengine_%s.h\"\n\n", use_iocp ? "iocp" : "select");\r
- fprintf(f, "#endif\n\n");\r
- fclose(f);\r
-\r
- sc(TGREEN); printf(" done\n"); sc(TNORMAL);\r
- printf("Writing command and module compilation scripts...");\r
- WriteCompileCommands();\r
- WriteCompileModules();\r
- sc(TGREEN); printf(" done\n"); sc(TNORMAL);\r
-\r
- printf("\nconfigure is done.. exiting!\n");\r
-}\r
-\r
-void WriteCompileCommands()\r
-{\r
- char commands[300][100];\r
- int command_count = 0;\r
- printf("\n Finding Command Sources...\n");\r
- WIN32_FIND_DATA fd;\r
- HANDLE fh = FindFirstFile("..\\src\\cmd_*.cpp", &fd);\r
- if(fh == INVALID_HANDLE_VALUE)\r
- printf_c("\033[1;32m No command sources could be found! This \033[1m*could*\033[1;32m be a bad thing.. :P\033[0m");\r
- else\r
- {\r
- sc(TGREEN);\r
- do \r
- {\r
- strcpy(commands[command_count], fd.cFileName);\r
- commands[command_count][strlen(fd.cFileName) - 4] = 0;\r
- printf(" %s\n", commands[command_count]);\r
- ++command_count;\r
- } while(FindNextFile(fh, &fd));\r
- sc(TNORMAL);\r
- }\r
- \r
- // Write our spiffy new makefile :D\r
- // I am such a lazy fucker :P\r
-#ifdef _DEBUG\r
- FILE * f = fopen("..\\src\\commands.mak", "w");\r
-#else\r
- FILE * f = fopen("..\\src\\commands-release.mak", "w");\r
-#endif\r
-\r
- fprintf(f, "# Generated at SOMETIME\n");\r
- fprintf(f, "!include <win32.mak>\n\n");\r
- fprintf(f, "all: ");\r
-\r
- // dump modules.. first time :)\r
- for(int i = 0; i < command_count; ++i)\r
- fprintf(f, "%s.so ", commands[i]);\r
-\r
- fprintf(f, "\n.cpp.obj:\n");\r
-#ifdef _DEBUG\r
- fprintf(f, " $(cc) /nologo /LD /Od /I \".\" /I \"../include\" /I \"../include/modes\" /I \"../include/commands\" /I \"../win\" /D \"WIN32\" /D \"_DEBUG\" /D \"_CONSOLE\" /D \"_MBCS\" /D \"DLL_BUILD\" /Gm /EHsc /GS /RTC1 /MTd /Fo\"Debug/\" /Fd\"Debug/vc70.pdb\" /W3 /Wp64 /Zi /TP $*.cpp ..\\win\\inspircd_memory_functions.cpp /link ..\\bin\\debug\\bin\\inspircd.lib /OUT:\"$*.so\"\n\n");\r
-#else\r
- fprintf(f, " $(cc) /nologo /LD /Od /I \".\" /I \"../include\" /I \"../include/modes\" /I \"../include/commands\" /I \"../win\" /D \"WIN32\" /D \"_CONSOLE\" /D \"_MBCS\" /D \"DLL_BUILD\" /EHsc /GS /MT /Fo\"Release/\" /Fd\"Release/vc70.pdb\" /W3 /Wp64 /Zi /TP $*.cpp ..\\win\\inspircd_memory_functions.cpp /link ..\\bin\\release\\bin\\inspircd.lib /OUT:\"$*.so\"\n\n");\r
-#endif\r
-\r
- \r
- // dump modules.. again the second and last time :)\r
- for(int i = 0; i < command_count; ++i)\r
- fprintf(f, "%s.so : %s.obj\n", commands[i], commands[i]);\r
-\r
- fprintf(f, "\n");\r
- fclose(f);\r
-}\r
-\r
-void WriteCompileModules()\r
-{\r
- char modules[300][100];\r
- int module_count = 0;\r
-\r
- printf("Finding Modules...\n");\r
- WIN32_FIND_DATA fd;\r
- HANDLE fh = FindFirstFile("..\\src\\modules\\m_*.cpp", &fd);\r
- if(fh == INVALID_HANDLE_VALUE)\r
- printf_c("\033[1;32m No module sources could be found! This \033[1m*could*\033[1;32m be a bad thing.. :P\033[0m");\r
- else\r
- {\r
- sc(TGREEN);\r
- do \r
- {\r
- strcpy(modules[module_count], fd.cFileName);\r
- modules[module_count][strlen(fd.cFileName) - 4] = 0;\r
- printf(" %s\n", modules[module_count]);\r
- ++module_count;\r
- } while(FindNextFile(fh, &fd));\r
- sc(TNORMAL);\r
- }\r
-\r
- // Write our spiffy new makefile :D\r
- // I am such a lazy fucker :P\r
-#ifdef _DEBUG\r
- FILE * f = fopen("..\\src\\modules\\modules.mak", "w");\r
-#else\r
- FILE * f = fopen("..\\src\\modules\\modules-release.mak", "w");\r
-#endif\r
-\r
- fprintf(f, "# Generated at SOMETIME\n");\r
- fprintf(f, "!include <win32.mak>\n\n");\r
- fprintf(f, "all: ");\r
-\r
- // dump modules.. first time :)\r
- for(int i = 0; i < module_count; ++i)\r
- fprintf(f, "%s.so ", modules[i]);\r
-\r
- fprintf(f, "\n.cpp.obj:\n");\r
-#ifdef _DEBUG\r
- fprintf(f, " $(cc) /nologo /LD /Od /I \".\" /I \"../../include\" /I \"../../include/modes\" /I \"../../include/modules\" /I \"../../win\" /D \"WIN32\" /D \"_DEBUG\" /D \"_CONSOLE\" /D \"_MBCS\" /D \"DLL_BUILD\" /Gm /EHsc /GS /RTC1 /MTd /Fo\"Debug/\" /Fd\"Debug/vc70.pdb\" /W3 /Wp64 /Zi /TP $*.cpp ..\\..\\win\\inspircd_memory_functions.cpp /link ..\\..\\bin\\debug\\bin\\inspircd.lib ws2_32.lib /OUT:\"$*.so\"\n\n");\r
-#else\r
- fprintf(f, " $(cc) /nologo /LD /Od /I \".\" /I \"../../include\" /I \"../../include/modes\" /I \"../../include/modules\" /I \"../../win\" /D \"WIN32\" /D \"_CONSOLE\" /D \"_MBCS\" /D \"DLL_BUILD\" /EHsc /GS /MT /Fo\"Release/\" /Fd\"Release/vc70.pdb\" /W3 /Wp64 /Zi /TP $*.cpp ..\\..\\win\\inspircd_memory_functions.cpp /link ..\\..\\bin\\release\\bin\\inspircd.lib ws2_32.lib /OUT:\"$*.so\"\n\n");\r
-#endif\r
-\r
-\r
- // dump modules.. again the second and last time :)\r
- for(int i = 0; i < module_count; ++i)\r
- fprintf(f, "%s.so : %s.obj\n", modules[i], modules[i]);\r
-\r
- fprintf(f, "\n");\r
- fclose(f);\r
-}\r
+/* +------------------------------------+
+ * | Inspire Internet Relay Chat Daemon |
+ * +------------------------------------+
+ *
+ * InspIRCd: (C) 2002-2011 InspIRCd Development Team
+ * See: http://wiki.inspircd.org/Credits
+ *
+ * This program is free but copyrighted software; see
+ * the file COPYING for details.
+ *
+ * ---------------------------------------------------
+ */
+
+#define _CRT_SECURE_NO_DEPRECATE
+
+#define CONFIGURE_BUILD
+#define WIN32_LEAN_AND_MEAN
+#include <windows.h>
+#include <stdio.h>
+#include <process.h>
+#include <iostream>
+#include <string>
+#include <vector>
+#include <time.h>
+#include "inspircd_win32wrapper.h"
+#include "colours.h"
+
+using namespace std;
+void Run();
+void Banner();
+void WriteCompileModules(const vector<string> &, const vector<string> &);
+void WriteCompileCommands();
+void Rebase();
+void CopyExtras();
+
+/* detects if we are running windows xp or higher (5.1) */
+bool iswinxp()
+{
+ OSVERSIONINFO vi;
+ vi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
+ GetVersionEx(&vi);
+ if(vi.dwMajorVersion >= 5)
+ return true;
+
+ return false;
+}
+
+int get_int_option(const char * text, int def)
+{
+ static char buffer[500];
+ int ret;
+ printf_c("%s\n[\033[1;32m%u\033[0m] -> ", text, def);
+ fgets(buffer, 500, stdin);
+ if(sscanf(buffer, "%u", &ret) != 1)
+ ret = def;
+
+ printf("\n");
+ return ret;
+}
+
+bool get_bool_option(const char * text, bool def)
+{
+ static char buffer[500];
+ char ret[100];
+ printf_c("%s [\033[1;32m%c\033[0m] -> ", text, def ? 'y' : 'n');
+ fgets(buffer, 500, stdin);
+ if(sscanf(buffer, "%s", ret) != 1)
+ strcpy(ret, def ? "y" : "n");
+
+ printf("\n");
+ return !strncmp(ret, "y", 1);
+}
+
+string get_string_option(const char * text, char * def)
+{
+ if (def && *def)
+ printf_c("%s\n[\033[1;32m%s\033[0m] -> ", text, def);
+ else
+ printf_c("%s\n[] -> ", text);
+
+ char buffer[1000], buf[1000];
+ fgets(buffer, sizeof(buffer), stdin);
+ if (sscanf(buffer, "%s", buf) != 1)
+ strcpy(buf, def);
+
+ printf("\n");
+ return buf;
+}
+
+// escapes a string for use in a c++ file
+void escape_string(string &str)
+{
+ string copy = str;
+ str.clear();
+
+ for (unsigned i = 0; i < copy.size(); ++i)
+ {
+ str += copy[i];
+ if (copy[i] == '\\')
+ str += '\\';
+ }
+}
+
+string get_git_commit()
+{
+ char buf[128];
+ char *ref = NULL, *commit = NULL;
+ FILE *f = fopen("../.git/HEAD", "r");
+ if (f)
+ {
+ if (fgets(buf, sizeof(buf), f))
+ {
+ while (isspace(buf[strlen(buf) - 1]))
+ buf[strlen(buf) - 1] = 0;
+ char *p = strchr(buf, ' ');
+ if (p)
+ ref = ++p;
+ }
+ fclose(f);
+ }
+ if (ref == NULL)
+ return "";
+ string ref_file = string("../.git/") + string(ref);
+ f = fopen(ref_file.c_str(), "r");
+ if (f)
+ {
+ if (fgets(buf, sizeof(buf), f))
+ {
+ while (isspace(buf[strlen(buf) - 1]))
+ buf[strlen(buf) - 1] = 0;
+ commit = buf;
+ }
+ fclose(f);
+ }
+
+ return commit != NULL ? commit : "";
+}
+
+void get_machine_info(char * buffer, size_t len)
+{
+ char buf[500];
+ char buf2[500];
+
+ DWORD dwSize = sizeof(buf);
+ if (!GetComputerNameEx((COMPUTER_NAME_FORMAT)ComputerNameDnsFullyQualified, buf, &dwSize))
+ sprintf(buf, "%s", "unknown");
+
+ FILE * f = fopen("ver.txt.tmp", "r");
+ if (f)
+ {
+ while (fgets(buf2, 500, f)) { }
+ fclose(f);
+ unlink("ver.txt.tmp");
+ }
+ else
+ sprintf(buf2, "%s", "unknown");
+
+ sprintf(buffer, "%s ", buf);
+ //strip newlines
+ char* b = buffer + strlen(buf)+1;
+ char *b2 = buf2;
+ while (*b2)
+ {
+ if (*b2 != 10 && *b2 != 13)
+ {
+ *b = *b2;
+ b++;
+ }
+ *b2++;
+ }
+ *b = 0;
+}
+
+vector<string> get_dir_list(const string &path_list)
+{
+ char *paths = strdup(path_list.c_str());
+ char *paths_save = paths;
+ char *p = paths;
+ vector<string> paths_return;
+
+ while ((p = strchr(paths, ';')))
+ {
+ *p++ = 0;
+ paths_return.push_back(paths);
+ paths = p;
+ }
+ if (paths != NULL)
+ paths_return.push_back(paths);
+ free(paths_save);
+
+ return paths_return;
+}
+
+int __stdcall WinMain(IN HINSTANCE hInstance, IN HINSTANCE hPrevInstance, IN LPSTR lpCmdLine, IN int nShowCmd )
+{
+ if (!strcmp(lpCmdLine, "/rebase"))
+ {
+ Rebase();
+ return 0;
+ }
+
+ FILE * j = fopen("inspircd_config.h", "r");
+ if (j)
+ {
+ if (MessageBox(0, "inspircd_config.h already exists. Remove it and build from clean?", "Configure program", MB_ICONQUESTION|MB_YESNO|MB_DEFBUTTON2) != IDYES)
+ {
+ fclose(j);
+ exit(0);
+ }
+ }
+
+ // call before we hook console handles
+ system("ver > ver.txt.tmp");
+
+ AllocConsole();
+
+ // pipe standard handles to this console
+ freopen("CONIN$", "r", stdin);
+ freopen("CONOUT$", "w", stdout);
+ freopen("CONOUT$", "w", stderr);
+
+ Banner();
+ Run();
+ FreeConsole();
+ return 0;
+}
+
+void Banner()
+{
+ printf_c("\nWelcome to the \033[1mInspIRCd\033[0m Configuration program! (\033[1minteractive mode\033[0m)\n"
+ "\033[1mPackage maintainers: Type ./configure --help for non-interactive help\033[0m\n\n");
+ printf_c("*** If you are unsure of any of these values, leave it blank for ***\n"
+ "*** standard settings that will work, and your server will run ***\n"
+ "*** using them. Please consult your IRC network admin if in doubt. ***\n\n"
+ "Press \033[1m<RETURN>\033[0m to accept the default for any option, or enter\n"
+ "a new value. Please note: You will \033[1mHAVE\033[0m to read the docs\n"
+ "dir, otherwise you won't have a config file!\n\n");
+
+}
+
+void Run()
+{
+ vector<string> extra_include_paths, extra_lib_paths;
+ string revision = get_git_commit();
+ char version[514];
+ char machine_text[MAX_PATH];
+ get_machine_info(machine_text, MAX_PATH);
+
+ // grab version
+ FILE * fI = fopen("..\\src\\version.sh", "r");
+ if(fI)
+ {
+ fgets(version, 514, fI);
+ fgets(version, 514, fI);
+ char * p2 = version;
+ while(*p2 != '\"')
+ ++p2;
+ ++p2;
+ strcpy(version, p2);
+ p2 = version;
+ while(*p2 != '\"')
+ ++p2;
+ *p2 = 0;
+ fclose(fI);
+ }
+ else
+ strcpy(version, "InspIRCD-Unknown");
+#ifdef WIN64
+ printf_c("Your operating system is: \033[1;32mwindows_x64 \033[0m\n");
+#else
+ printf_c("Your operating system is: \033[1;32mwindows_x32 \033[0m\n");
+#endif
+ printf_c("InspIRCd revision ID: \033[1;32m%s \033[0m\n\n", !revision.empty() ? revision.c_str() : "(Non-GIT build)");
+
+ printf_c("\033[1mExtra modules.\033[0m\n");
+ if (get_bool_option("Do you want to compile any extra non-core modules?", false))
+ {
+ string extra_i_path = get_string_option("Extra include search paths separate by \";\"", ".");
+ string extra_l_path = get_string_option("Extra library search paths, separate by \";\"", ".");
+
+ extra_include_paths = get_dir_list(extra_i_path);
+ extra_lib_paths = get_dir_list(extra_l_path);
+ }
+
+ printf_c("\033[1mAll paths are relative to the binary directory.\033[0m\n");
+ string base_path = get_string_option("In what directory do you wish to install the InspIRCd base?", "..");
+ string config_file = get_string_option("In what directory are the configuration files?", "conf");
+ string mod_path = get_string_option("In what directory are the modules to be compiled to?", "modules");
+ string bin_dir = get_string_option("In what directory is the IRCd binary to be placed?", ".");
+
+ printf_c("\n\033[1;32mPre-build configuration is complete!\n\n"); sc(TNORMAL);
+
+ CopyExtras();
+
+ // dump all the options back out
+ printf_c("\033[0mBase install path:\033[1;32m %s\n", base_path.c_str());
+ printf_c("\033[0mConfig path:\033[1;32m %s\n", config_file.c_str());
+ printf_c("\033[0mModule path:\033[1;32m %s\n", mod_path.c_str());
+ printf_c("\033[0mSocket Engine:\033[1;32m %s\n", "select");
+
+ printf("\n"); sc(TNORMAL);
+ if(get_bool_option("Are these settings correct?", true) == false)
+ {
+ Run();
+ return;
+ }
+ printf("\n");
+
+ // escape the pathes
+ escape_string(config_file);
+ escape_string(mod_path);
+
+ printf("\nWriting inspircd_config.h...");
+ FILE * f = fopen("inspircd_config.h", "w");
+ fprintf(f, "/* Auto generated by configure, do not modify! */\n");
+ fprintf(f, "#ifndef __CONFIGURATION_AUTO__\n");
+ fprintf(f, "#define __CONFIGURATION_AUTO__\n\n");
+
+ fprintf(f, "#define MOD_PATH \"%s\"\n", mod_path.c_str());
+ fprintf(f, "#define SOMAXCONN_S \"128\"\n");
+ fprintf(f, "#define MAXBUF 514\n");
+
+ fprintf(f, "\n#include \"inspircd_win32wrapper.h\"");
+ fprintf(f, "\n#include \"inspircd_namedpipe.h\"");
+ fprintf(f, "\n#include \"threadengines/threadengine_win32.h\"\n\n");
+ fprintf(f, "#endif\n\n");
+ fclose(f);
+
+ sc(TGREEN); printf(" done\n"); sc(TNORMAL);
+ printf("Writing inspircd_se_config.h...");
+
+ f = fopen("inspircd_se_config.h", "w");
+ fprintf(f, "/* Auto generated by configure, do not modify or commit to svn! */\n");
+ fprintf(f, "#ifndef __CONFIGURATION_SOCKETENGINE__\n");
+ fprintf(f, "#define __CONFIGURATION_SOCKETENGINE__\n\n");
+ fprintf(f, "#include \"socketengines/socketengine_%s.h\"\n\n", "select");
+ fprintf(f, "#endif\n\n");
+ fclose(f);
+
+ sc(TGREEN); printf(" done\n"); sc(TNORMAL);
+ printf("Writing inspircd_version.h...");
+ f = fopen("inspircd_version.h", "w");
+ fprintf(f, "#define VERSION \"%s\"\n", version);
+ fprintf(f, "#define REVISION \"%s\"\n", revision.c_str());
+ fprintf(f, "#define SYSTEM \"%s\"\n", machine_text);
+ fclose(f);
+
+ sc(TGREEN); printf(" done\n"); sc(TNORMAL);
+ printf("Writing command and module compilation scripts...");
+ WriteCompileCommands();
+ WriteCompileModules(extra_include_paths, extra_lib_paths);
+ sc(TGREEN); printf(" done\n"); sc(TNORMAL);
+
+ printf("\nconfigure is done.. exiting!\n");
+}
+
+/* Keeps files from modules/extra up to date if theyre copied into modules/ */
+void CopyExtras()
+{
+ char dest[65535];
+ char src[65535];
+
+ printf("\nUpdating extra modules in src/modules...\n");
+
+ WIN32_FIND_DATA fd;
+ HANDLE fh = FindFirstFile("..\\src\\modules\\extra\\*.*", &fd);
+
+ if(fh == INVALID_HANDLE_VALUE)
+ return;
+
+ do
+ {
+ strcpy(dest, "..\\src\\modules\\");
+ strcat(dest, fd.cFileName);
+ strcpy(src, "..\\src\\modules\\extra\\");
+ strcat(src, fd.cFileName);
+ FILE* x = fopen(dest, "r");
+ if (x)
+ {
+ fclose(x);
+ CopyFile(src, dest, false);
+ sc(TGREEN); printf(" %s", fd.cFileName); sc(TNORMAL);
+ printf("...\n");
+ }
+ }
+ while (FindNextFile(fh, &fd));
+
+ FindClose(fh);
+
+ printf("\n\n");
+}
+
+
+void Rebase()
+{
+ char dest[65535];
+ char command[65535];
+
+ WIN32_FIND_DATA fd;
+
+#ifdef _DEBUG
+ HANDLE fh = FindFirstFile("..\\bin\\debug\\modules\\*.so", &fd);
+#else
+ HANDLE fh = FindFirstFile("..\\bin\\release\\modules\\*.so", &fd);
+#endif
+ if(fh == INVALID_HANDLE_VALUE)
+ return;
+
+ *dest = 0;
+
+ do
+ {
+#ifdef _DEBUG
+ strcat(dest, " ..\\bin\\debug\\modules\\");
+#else
+ strcat(dest, " ..\\bin\\release\\modules\\");
+#endif
+ strcat(dest, fd.cFileName);
+ }
+ while (FindNextFile(fh, &fd));
+
+ sprintf(command, "rebase.exe -v -b 11000000 -c baseaddr_modules.txt %s", dest);
+ printf("%s\n", command);
+ system(command);
+
+ FindClose(fh);
+}
+
+void WriteCompileCommands()
+{
+ char commands[300][100];
+ int command_count = 0;
+ printf("\n Finding Command Sources...\n");
+ WIN32_FIND_DATA fd;
+ HANDLE fh = FindFirstFile("..\\src\\commands\\cmd_*.cpp", &fd);
+ if(fh == INVALID_HANDLE_VALUE)
+ printf_c("\033[1;32m No command sources could be found! This \033[1m*could*\033[1;32m be a bad thing.. :P\033[0m");
+ else
+ {
+ sc(TGREEN);
+ do
+ {
+ strcpy(commands[command_count], fd.cFileName);
+ commands[command_count][strlen(fd.cFileName) - 4] = 0;
+ printf(" %s\n", commands[command_count]);
+ ++command_count;
+ } while(FindNextFile(fh, &fd));
+ sc(TNORMAL);
+ }
+
+ // Write our spiffy new makefile :D
+ // I am such a lazy fucker :P
+ FILE * f = fopen("..\\src\\commands\\commands.mak", "w");
+
+ time_t t = time(NULL);
+ fprintf(f, "# Generated at %s\n", ctime(&t));
+ fprintf(f, "all: makedir ");
+
+ // dump modules.. first time :)
+ for(int i = 0; i < command_count; ++i)
+ fprintf(f, "%s.so ", commands[i]);
+
+ fprintf(f, "\n.cpp.obj:\n");
+#ifdef WIN64
+ // /MACHINE:X64
+ #ifdef _DEBUG
+ fprintf(f, " cl /nologo /LD /Od /I \".\" /I \"../../include\" /I \"../../include/modes\" /I \"../../include/commands\" /I \"../../win\" /D \"WIN32\" /D \"_DEBUG\" /D \"_CONSOLE\" /D \"_MBCS\" /D \"DLL_BUILD\" /Gm /EHsc /RTC1 /MDd /Fo\"Debug/\" /Fd\"Debug/vc90.pdb\" /W2 /Zi /TP $*.cpp ..\\..\\win\\inspircd_memory_functions.cpp /link ..\\..\\bin\\debug_x64\\inspircd.lib /OUT:\"..\\..\\bin\\debug_x64\\modules\\$*.so\" /PDB:\"..\\..\\bin\\debug_x64\\modules\\$*.pdb\" /MACHINE:X64 /IMPLIB:\"..\\..\\bin\\debug_x64\\modules\\$*.lib\"\n\n");
+ CreateDirectory("..\\src\\commands\\debug", NULL);
+ CreateDirectory("..\\bin\\debug\\modules", NULL);
+ #else
+ fprintf(f, " cl /nologo /LD /Od /I \".\" /I \"../../include\" /I \"../../include/modes\" /I \"../../include/commands\" /I \"../../win\" /D \"WIN32\" /D \"_CONSOLE\" /D \"_MBCS\" /D \"DLL_BUILD\" /Gm /EHsc /GL /MD /Fo\"Release/\" /Fd\"Release/vc90.pdb\" /W2 /Zi /TP $*.cpp ..\\..\\win\\inspircd_memory_functions.cpp /link ..\\..\\bin\\release_x64\\inspircd.lib /OUT:\"..\\..\\bin\\release_x64\\modules\\$*.so\" /PDB:\"..\\..\\bin\\release_x64\\modules\\$*.pdb\" /MACHINE:X64 /IMPLIB:\"..\\..\\bin\\release_x64\\modules\\$*.lib\"\n\n");
+ CreateDirectory("..\\src\\commands\\release", NULL);
+ CreateDirectory("..\\bin\\release\\modules", NULL);
+ #endif
+#else
+ #ifdef _DEBUG
+ fprintf(f, " cl /nologo /LD /Od /I \".\" /I \"../../include\" /I \"../../include/modes\" /I \"../../include/commands\" /I \"../../win\" /D \"WIN32\" /D \"_DEBUG\" /D \"_CONSOLE\" /D \"_MBCS\" /D \"DLL_BUILD\" /Gm /EHsc /RTC1 /MDd /Fo\"Debug/\" /Fd\"Debug/vc90.pdb\" /W2 /Zi /TP $*.cpp ..\\..\\win\\inspircd_memory_functions.cpp /link ..\\..\\bin\\debug\\inspircd.lib /OUT:\"..\\..\\bin\\debug\\modules\\$*.so\" /PDB:\"..\\..\\bin\\debug\\modules\\$*.pdb\" /IMPLIB:\"..\\..\\bin\\debug\\modules\\$*.lib\"\n\n");
+ CreateDirectory("..\\src\\commands\\debug", NULL);
+ CreateDirectory("..\\bin\\debug\\modules", NULL);
+ #else
+ fprintf(f, " cl /nologo /LD /Od /I \".\" /I \"../../include\" /I \"../../include/modes\" /I \"../../include/commands\" /I \"../../win\" /D \"WIN32\" /D \"_CONSOLE\" /D \"_MBCS\" /D \"DLL_BUILD\" /Gm /EHsc /GL /MD /Fo\"Release/\" /Fd\"Release/vc90.pdb\" /W2 /Zi /TP $*.cpp ..\\..\\win\\inspircd_memory_functions.cpp /link ..\\..\\bin\\release\\inspircd.lib /OUT:\"..\\..\\bin\\release\\modules\\$*.so\" /PDB:\"..\\..\\bin\\release\\modules\\$*.pdb\" /IMPLIB:\"..\\..\\bin\\release\\modules\\$*.lib\"\n\n");
+ CreateDirectory("..\\src\\commands\\release", NULL);
+ CreateDirectory("..\\bin\\release\\modules", NULL);
+ #endif
+#endif
+
+ fprintf(f, "makedir:\n");
+#ifdef _DEBUG
+ fprintf(f, " if not exist ..\\..\\bin\\debug mkdir ..\\..\\bin\\debug\n");
+ fprintf(f, " if not exist ..\\..\\bin\\debug\\modules mkdir ..\\..\\bin\\debug\\modules\n");
+ fprintf(f, " if not exist ..\\..\\bin\\debug\\data mkdir ..\\..\\bin\\debug\\data\n");
+ fprintf(f, " if not exist ..\\..\\bin\\debug\\logs mkdir ..\\..\\bin\\debug\\logs\n");
+#else
+ fprintf(f, " if not exist ..\\..\\bin\\release mkdir ..\\..\\bin\\release\n");
+ fprintf(f, " if not exist ..\\..\\bin\\release\\modules mkdir ..\\..\\bin\\release\\modules\n");
+ fprintf(f, " if not exist ..\\..\\bin\\release\\data mkdir ..\\..\\bin\\release\\data\n");
+ fprintf(f, " if not exist ..\\..\\bin\\release\\logs mkdir ..\\..\\bin\\release\\logs\n");
+#endif
+
+ // dump modules.. again the second and last time :)
+ for(int i = 0; i < command_count; ++i)
+ fprintf(f, "%s.so : %s.obj\n", commands[i], commands[i]);
+
+ fprintf(f, "\n");
+ fclose(f);
+}
+
+void WriteCompileModules(const vector<string> &includes, const vector<string> &libs)
+{
+ char modules[300][100];
+ int module_count = 0;
+
+ printf("Finding Modules...\n");
+ WIN32_FIND_DATA fd;
+ HANDLE fh = FindFirstFile("..\\src\\modules\\m_*.cpp", &fd);
+ if(fh == INVALID_HANDLE_VALUE)
+ printf_c("\033[1;32m No module sources could be found! This \033[1m*could*\033[1;32m be a bad thing.. :P\033[0m");
+ else
+ {
+ sc(TGREEN);
+ do
+ {
+ strcpy(modules[module_count], fd.cFileName);
+ modules[module_count][strlen(fd.cFileName) - 4] = 0;
+ printf(" %s\n", modules[module_count]);
+ ++module_count;
+ } while(FindNextFile(fh, &fd));
+ sc(TNORMAL);
+ }
+
+ string extra_include, extra_lib;
+ for (unsigned i = 0; i < includes.size(); ++i)
+ extra_include += " /I \"" + includes[i] + "\" ";
+ for (unsigned i = 0; i < libs.size(); ++i)
+ extra_lib += " /LIBPATH:\"" + libs[i] + "\" ";
+
+ // Write our spiffy new makefile :D
+ // I am such a lazy fucker :P
+ FILE * f = fopen("..\\src\\modules\\modules.mak", "w");
+
+ time_t t = time(NULL);
+ fprintf(f, "# Generated at %s\n", ctime(&t));
+ fprintf(f, "all: makedir ");
+
+ // dump modules.. first time :)
+ for(int i = 0; i < module_count; ++i)
+ fprintf(f, "%s.so ", modules[i]);
+
+ fprintf(f, "\n.cpp.obj:\n");
+#ifdef WIN64
+ // /MACHINE:X64
+ #ifdef _DEBUG
+ fprintf(f, " cl /nologo /LD /Od /I \".\" /I \"../../include\" /I \"../../include/modes\" /I \"../../include/modules\" /I \"../../win\" %s /D \"WIN32\" /D \"_DEBUG\" /D \"_CONSOLE\" /D \"_MBCS\" /D \"DLL_BUILD\" /Gm /EHsc /RTC1 /MDd /Fo\"Debug/\" /Fd\"Debug/vc90.pdb\" /W2 /Zi /TP $*.cpp ..\\..\\win\\inspircd_memory_functions.cpp /link %s ..\\..\\bin\\debug_x64\\inspircd.lib ws2_32.lib /OUT:\"..\\..\\bin\\debug_x64\\modules\\$*.so\" /PDB:\"..\\..\\bin\\debug_x64\\modules\\$*.pdb\" /MACHINE:X64 /IMPLIB:\"..\\..\\bin\\debug_x64\\modules\\$*.lib\"\n\n", extra_include.c_str(), extra_lib.c_str());
+ CreateDirectory("..\\src\\modules\\debug_x64", NULL);
+ #else
+ fprintf(f, " cl /nologo /LD /Od /I \".\" /I \"../../include\" /I \"../../include/modes\" /I \"../../include/modules\" /I \"../../win\" %s /D \"WIN32\" /D \"_CONSOLE\" /D \"_MBCS\" /D \"DLL_BUILD\" /Gm /EHsc /GL /MD /Fo\"Release/\" /Fd\"Release/vc90.pdb\" /W2 /Zi /TP $*.cpp ..\\..\\win\\inspircd_memory_functions.cpp /link %s ..\\..\\bin\\release_x64\\inspircd.lib ws2_32.lib /OUT:\"..\\..\\bin\\release_x64\\modules\\$*.so\" /PDB:\"..\\..\\bin\\release_x64\\modules\\$*.pdb\" /MACHINE:X64 /IMPLIB:\"..\\..\\bin\\release_x64\\modules\\$*.lib\"\n\n", extra_include.c_str(), extra_lib.c_str());
+ CreateDirectory("..\\src\\modules\\release_x64", NULL);
+ #endif
+#else
+ #ifdef _DEBUG
+ fprintf(f, " cl /nologo /LD /Od /I \".\" /I \"../../include\" /I \"../../include/modes\" /I \"../../include/modules\" /I \"../../win\" %s /D \"WIN32\" /D \"_DEBUG\" /D \"_CONSOLE\" /D \"_MBCS\" /D \"DLL_BUILD\" /Gm /EHsc /RTC1 /MDd /Fo\"Debug/\" /Fd\"Debug/vc90.pdb\" /W2 /Zi /TP $*.cpp ..\\..\\win\\inspircd_memory_functions.cpp /link %s ..\\..\\bin\\debug\\inspircd.lib ws2_32.lib /OUT:\"..\\..\\bin\\debug\\modules\\$*.so\" /PDB:\"..\\..\\bin\\debug\\modules\\$*.pdb\" /IMPLIB:\"..\\..\\bin\\debug\\modules\\$*.lib\"\n\n", extra_include.c_str(), extra_lib.c_str());
+ CreateDirectory("..\\src\\modules\\debug", NULL);
+ #else
+ fprintf(f, " cl /nologo /LD /Od /I \".\" /I \"../../include\" /I \"../../include/modes\" /I \"../../include/modules\" /I \"../../win\" %s /D \"WIN32\" /D \"_CONSOLE\" /D \"_MBCS\" /D \"DLL_BUILD\" /Gm /EHsc /GL /MD /Fo\"Release/\" /Fd\"Release/vc90.pdb\" /W2 /Zi /TP $*.cpp ..\\..\\win\\inspircd_memory_functions.cpp /link %s ..\\..\\bin\\release\\inspircd.lib ws2_32.lib /OUT:\"..\\..\\bin\\release\\modules\\$*.so\" /PDB:\"..\\..\\bin\\release\\modules\\$*.pdb\" /IMPLIB:\"..\\..\\bin\\release\\modules\\$*.lib\"\n\n", extra_include.c_str(), extra_lib.c_str());
+ CreateDirectory("..\\src\\modules\\release", NULL);
+ #endif
+#endif
+
+#ifdef _DEBUG
+ fprintf(f, "makedir:\n if not exist debug mkdir debug\n\n");
+#else
+ fprintf(f, "makedir:\n if not exist release mkdir release\n\n");
+#endif
+
+ // dump modules.. again the second and last time :)
+ for(int i = 0; i < module_count; ++i)
+ fprintf(f, "%s.so : %s.obj\n", modules[i], modules[i]);
+
+ fprintf(f, "\n");
+ fclose(f);
+}