2002-08-02 10:01:35 +00:00
|
|
|
#include <windows.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "exehead/config.h"
|
|
|
|
#include "exehead/fileform.h"
|
|
|
|
|
|
|
|
#include "exedata.h"
|
|
|
|
|
|
|
|
#include "build.h"
|
|
|
|
#include "util.h"
|
|
|
|
|
|
|
|
#include "ResourceEditor.h"
|
|
|
|
#include "exehead/resource.h"
|
2002-11-01 20:34:55 +00:00
|
|
|
#include "exehead/lang.h"
|
2002-08-02 10:01:35 +00:00
|
|
|
|
2002-08-31 23:44:58 +00:00
|
|
|
void CEXEBuild::define(const char *p, const char *v)
|
|
|
|
{
|
|
|
|
definedlist.add(p,v);
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CEXEBuild::~CEXEBuild()
|
|
|
|
{
|
|
|
|
free(header_data_new);
|
|
|
|
free(m_unicon_data);
|
2003-04-21 13:32:34 +00:00
|
|
|
for (unsigned int i = 0; i < build_nlfs.size(); i++)
|
|
|
|
delete build_nlfs[i];
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CEXEBuild::CEXEBuild()
|
|
|
|
{
|
|
|
|
display_info=1;
|
|
|
|
display_script=1;
|
|
|
|
display_errors=1;
|
|
|
|
display_warnings=1;
|
|
|
|
|
|
|
|
has_called_write_output=0;
|
|
|
|
|
|
|
|
ns_func.add("",0); // make sure offset 0 is special on these (i.e. never used by a label)
|
|
|
|
ns_label.add("",0);
|
|
|
|
|
|
|
|
header_data_new=(unsigned char*)malloc(zlib_exeheader_size);
|
|
|
|
exeheader_size_new=zlib_exeheader_size;
|
|
|
|
exeheader_size=zlib_exeheader_size;
|
|
|
|
|
2002-08-31 23:44:58 +00:00
|
|
|
if (!header_data_new)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
ERROR_MSG("Internal compiler error #12345: malloc(%d) failed\n",exeheader_size_new);
|
|
|
|
extern void quit(); quit();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Changed by Amir Szekely 31st July 2002
|
|
|
|
memcpy(header_data_new,zlib_header_data,zlib_exeheader_size);
|
|
|
|
|
|
|
|
#ifdef NSIS_CONFIG_UNINSTALL_SUPPORT
|
|
|
|
// Changed by Amir Szekely 11th July 2002
|
|
|
|
// No need to check for uninstaller icon if uninstall support is disabled.
|
|
|
|
if (unicondata_size != icondata_size)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Internal compiler error #12345: installer,uninstaller icon size mismatch (%d,%d)\n",icondata_size,unicondata_size);
|
|
|
|
extern void quit(); quit();
|
|
|
|
}
|
|
|
|
#endif // NSIS_CONFIG_UNINSTALL_SUPPORT
|
|
|
|
|
|
|
|
strcpy(cur_out_path,"$INSTDIR");
|
|
|
|
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_BZIP2_COMPRESS_WHOLE
|
|
|
|
definedlist.add("NSIS_BZIP2_COMPRESS_WHOLE");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
{
|
|
|
|
char bzip_level[32];
|
|
|
|
wsprintf(bzip_level, "%d", NSIS_COMPRESS_BZIP2_LEVEL);
|
|
|
|
definedlist.add("NSIS_COMPRESS_BZIP2_LEVEL", bzip_level);
|
|
|
|
}
|
|
|
|
#ifdef NSIS_CONFIG_COMPONENTPAGE
|
|
|
|
definedlist.add("NSIS_CONFIG_COMPONENTPAGE");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
|
|
|
#ifdef NSIS_CONFIG_COMPRESSION_SUPPORT
|
|
|
|
definedlist.add("NSIS_CONFIG_COMPRESSION_SUPPORT");
|
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_CONFIG_CRC_ANAL
|
|
|
|
definedlist.add("NSIS_CONFIG_CRC_ANAL");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
|
|
|
#ifdef NSIS_CONFIG_CRC_SUPPORT
|
|
|
|
definedlist.add("NSIS_CONFIG_CRC_SUPPORT");
|
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_CONFIG_ENHANCEDUI_SUPPORT
|
|
|
|
definedlist.add("NSIS_CONFIG_ENHANCEDUI_SUPPORT");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_CONFIG_LICENSEPAGE
|
|
|
|
definedlist.add("NSIS_CONFIG_LICENSEPAGE");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_CONFIG_LOG
|
|
|
|
definedlist.add("NSIS_CONFIG_LOG");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_CONFIG_PLUGIN_SUPPORT
|
|
|
|
definedlist.add("NSIS_CONFIG_PLUGIN_SUPPORT");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_CONFIG_SILENT_SUPPORT
|
|
|
|
definedlist.add("NSIS_CONFIG_SILENT_SUPPORT");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_CONFIG_UNINSTALL_SUPPORT
|
|
|
|
definedlist.add("NSIS_CONFIG_UNINSTALL_SUPPORT");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_CONFIG_VISIBLE_SUPPORT
|
|
|
|
definedlist.add("NSIS_CONFIG_VISIBLE_SUPPORT");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
{
|
|
|
|
char max_inst_types[32];
|
|
|
|
wsprintf(max_inst_types, "%d", NSIS_MAX_INST_TYPES);
|
|
|
|
definedlist.add("NSIS_MAX_INST_TYPES", max_inst_types);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
char max_strlen[32];
|
|
|
|
wsprintf(max_strlen, "%d", NSIS_MAX_STRLEN);
|
|
|
|
definedlist.add("NSIS_MAX_STRLEN", max_strlen);
|
|
|
|
}
|
|
|
|
#ifdef NSIS_SUPPORT_ACTIVEXREG
|
|
|
|
definedlist.add("NSIS_SUPPORT_ACTIVEXREG");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_SUPPORT_BGBG
|
|
|
|
definedlist.add("NSIS_SUPPORT_BGBG");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_SUPPORT_CODECALLBACKS
|
|
|
|
definedlist.add("NSIS_SUPPORT_CODECALLBACKS");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_SUPPORT_COPYFILES
|
|
|
|
definedlist.add("NSIS_SUPPORT_COPYFILES");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
|
|
|
#ifdef NSIS_SUPPORT_CREATESHORTCUT
|
|
|
|
definedlist.add("NSIS_SUPPORT_CREATESHORTCUT");
|
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_SUPPORT_DELETE
|
|
|
|
definedlist.add("NSIS_SUPPORT_DELETE");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_SUPPORT_ENVIRONMENT
|
|
|
|
definedlist.add("NSIS_SUPPORT_ENVIRONMENT");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
|
|
|
#ifdef NSIS_SUPPORT_EXECUTE
|
|
|
|
definedlist.add("NSIS_SUPPORT_EXECUTE");
|
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_SUPPORT_FILE
|
|
|
|
definedlist.add("NSIS_SUPPORT_FILE");
|
|
|
|
#endif
|
|
|
|
#ifdef NSIS_SUPPORT_FILEFUNCTIONS
|
|
|
|
definedlist.add("NSIS_SUPPORT_FILEFUNCTIONS");
|
|
|
|
#endif
|
|
|
|
#ifdef NSIS_SUPPORT_FINDFIRST
|
|
|
|
definedlist.add("NSIS_SUPPORT_FINDFIRST");
|
|
|
|
#endif
|
|
|
|
#ifdef NSIS_SUPPORT_FNUTIL
|
|
|
|
definedlist.add("NSIS_SUPPORT_FNUTIL");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
|
|
|
#ifdef NSIS_SUPPORT_GETDLLVERSION
|
|
|
|
definedlist.add("NSIS_SUPPORT_GETDLLVERSION");
|
|
|
|
#endif
|
|
|
|
#ifdef NSIS_SUPPORT_GETFILETIME
|
|
|
|
definedlist.add("NSIS_SUPPORT_GETFILETIME");
|
|
|
|
#endif
|
|
|
|
#ifdef NSIS_SUPPORT_HWNDS
|
|
|
|
definedlist.add("NSIS_SUPPORT_HWNDS");
|
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_SUPPORT_INIFILES
|
|
|
|
definedlist.add("NSIS_SUPPORT_INIFILES");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_SUPPORT_INTOPTS
|
|
|
|
definedlist.add("NSIS_SUPPORT_INTOPTS");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_SUPPORT_MESSAGEBOX
|
|
|
|
definedlist.add("NSIS_SUPPORT_MESSAGEBOX");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_SUPPORT_MOVEONREBOOT
|
|
|
|
definedlist.add("NSIS_SUPPORT_MOVEONREBOOT");
|
|
|
|
#endif
|
|
|
|
#ifdef NSIS_SUPPORT_REBOOT
|
|
|
|
definedlist.add("NSIS_SUPPORT_REBOOT");
|
|
|
|
#endif
|
|
|
|
#ifdef NSIS_SUPPORT_REGISTRYFUNCTIONS
|
|
|
|
definedlist.add("NSIS_SUPPORT_REGISTRYFUNCTIONS");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
|
|
|
#ifdef NSIS_SUPPORT_RENAME
|
|
|
|
definedlist.add("NSIS_SUPPORT_RENAME");
|
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_SUPPORT_RMDIR
|
|
|
|
definedlist.add("NSIS_SUPPORT_RMDIR");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_SUPPORT_SHELLEXECUTE
|
|
|
|
definedlist.add("NSIS_SUPPORT_SHELLEXECUTE");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
2002-12-05 19:28:45 +00:00
|
|
|
#ifdef NSIS_SUPPORT_STACK
|
|
|
|
definedlist.add("NSIS_SUPPORT_STACK");
|
|
|
|
#endif
|
|
|
|
#ifdef NSIS_SUPPORT_STROPTS
|
|
|
|
definedlist.add("NSIS_SUPPORT_STROPTS");
|
|
|
|
#endif
|
|
|
|
#ifdef NSIS_ZLIB_COMPRESS_WHOLE
|
|
|
|
definedlist.add("NSIS_ZLIB_COMPRESS_WHOLE");
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
// Added by Amir Szekely 11th July 2002
|
|
|
|
// Coded by Robert Rainwater
|
|
|
|
{
|
|
|
|
char szNSISDir[NSIS_MAX_STRLEN],*fn2;
|
2002-12-15 20:54:17 +00:00
|
|
|
GetModuleFileName(NULL,szNSISDir,sizeof(szNSISDir)-sizeof("\\Include"));
|
2002-08-02 10:01:35 +00:00
|
|
|
fn2=strrchr(szNSISDir,'\\');
|
|
|
|
if(fn2!=NULL) *fn2=0;
|
|
|
|
definedlist.add("NSISDIR",(char*)szNSISDir);
|
2002-12-15 20:54:17 +00:00
|
|
|
lstrcat(szNSISDir, "\\Include");
|
|
|
|
include_dirs.add(szNSISDir,0);
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
db_opt_save=db_comp_save=db_full_size=db_opt_save_u=db_comp_save_u=db_full_size_u=0;
|
|
|
|
|
|
|
|
// Added by Amir Szekely 31st July 2002
|
2002-09-18 19:08:53 +00:00
|
|
|
#ifdef NSIS_CONFIG_COMPRESSION_SUPPORT
|
2002-08-21 16:23:29 +00:00
|
|
|
compressor = &zlib_compressor;
|
2002-09-18 19:08:53 +00:00
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
build_compressor_set = false;
|
|
|
|
#ifdef NSIS_ZLIB_COMPRESS_WHOLE
|
|
|
|
build_compress_whole = true;
|
|
|
|
#else
|
|
|
|
build_compress_whole = false;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
cur_entries=&build_entries;
|
|
|
|
cur_datablock=&build_datablock;
|
|
|
|
cur_functions=&build_functions;
|
|
|
|
cur_labels=&build_labels;
|
|
|
|
|
|
|
|
subsection_open_cnt=0;
|
|
|
|
build_cursection_isfunc=0;
|
|
|
|
build_cursection=NULL;
|
|
|
|
// init public data.
|
|
|
|
build_packname[0]=build_packcmd[0]=build_output_filename[0]=0;
|
|
|
|
|
|
|
|
build_overwrite=0;
|
|
|
|
build_compress=1;
|
2002-08-31 23:44:58 +00:00
|
|
|
build_crcchk=1;
|
2002-08-02 10:01:35 +00:00
|
|
|
build_datesave=1;
|
|
|
|
build_optimize_datablock=1;
|
|
|
|
|
|
|
|
memset(&build_header,-1,sizeof(build_header));
|
|
|
|
|
|
|
|
build_header.install_reg_rootkey=0;
|
|
|
|
build_header.num_sections=0;
|
|
|
|
build_header.common.num_entries=0;
|
2003-03-18 20:36:52 +00:00
|
|
|
build_header.common.flags=CH_FLAGS_NO_ROOT_DIR;
|
2002-08-02 10:01:35 +00:00
|
|
|
build_header.common.lb_bg=RGB(0,0,0);
|
|
|
|
build_header.common.lb_fg=RGB(0,255,0);
|
|
|
|
|
|
|
|
uninstall_mode=0;
|
|
|
|
uninstall_size_full=0;
|
|
|
|
uninstall_size=-1;
|
|
|
|
|
|
|
|
memset(&build_uninst,-1,sizeof(build_uninst));
|
|
|
|
build_uninst.common.lb_bg=RGB(0,0,0);
|
|
|
|
build_uninst.common.lb_fg=RGB(0,255,0);
|
|
|
|
build_uninst.common.num_entries=0;
|
|
|
|
build_uninst.code=0;
|
|
|
|
build_uninst.code_size=-1;
|
2003-03-20 13:13:48 +00:00
|
|
|
build_uninst.common.flags=0;
|
2002-08-02 10:01:35 +00:00
|
|
|
|
|
|
|
uninstaller_writes_used=0;
|
|
|
|
|
2002-09-29 20:25:15 +00:00
|
|
|
build_strlist.add("",0);
|
|
|
|
ubuild_strlist.add("",0);
|
|
|
|
build_header.install_directory_ptr=0;
|
|
|
|
build_header.install_reg_key_ptr=0;
|
2002-10-04 14:45:20 +00:00
|
|
|
#ifdef NSIS_CONFIG_COMPONENTPAGE
|
2003-05-24 13:50:24 +00:00
|
|
|
memset(build_header.install_types,0,sizeof(build_header.install_types));
|
2002-10-04 14:45:20 +00:00
|
|
|
#endif
|
2002-09-29 20:25:15 +00:00
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
// Changed by Amir Szekely 11th July 2002
|
|
|
|
// Changed to fit the new format in which uninstaller icons are saved
|
|
|
|
m_unicon_data=(unsigned char *)malloc(unicondata_size+3*sizeof(DWORD));
|
2002-09-23 14:07:39 +00:00
|
|
|
memcpy(m_unicon_data+2*sizeof(DWORD),unicon_data+22,unicondata_size);
|
2002-09-23 23:29:20 +00:00
|
|
|
*(DWORD*)(m_unicon_data) = unicondata_size;
|
|
|
|
*(DWORD*)(m_unicon_data + sizeof(DWORD)) = 0;
|
|
|
|
*(DWORD*)(m_unicon_data + 2*sizeof(DWORD) + unicondata_size) = 0;
|
2002-08-02 10:01:35 +00:00
|
|
|
unicondata_size += 3*sizeof(DWORD);
|
|
|
|
|
|
|
|
m_inst_fileused=0;
|
|
|
|
m_uninst_fileused=0;
|
|
|
|
|
2003-03-26 17:47:46 +00:00
|
|
|
branding_image_found=false;
|
2002-08-31 23:44:58 +00:00
|
|
|
|
2002-08-03 23:06:10 +00:00
|
|
|
no_space_texts=false;
|
2002-08-21 16:23:29 +00:00
|
|
|
|
2002-11-01 20:34:55 +00:00
|
|
|
#ifdef NSIS_CONFIG_PLUGIN_SUPPORT
|
2002-10-04 22:54:49 +00:00
|
|
|
build_plugin_unload=0;
|
2002-11-01 20:34:55 +00:00
|
|
|
#endif
|
2002-10-04 22:54:49 +00:00
|
|
|
|
2002-08-21 16:23:29 +00:00
|
|
|
last_used_lang=MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US);
|
2002-11-01 20:34:55 +00:00
|
|
|
|
|
|
|
build_header.common.num_pages=0;
|
|
|
|
#ifdef NSIS_CONFIG_UNINSTALL_SUPPORT
|
|
|
|
build_uninst.common.num_pages=0;
|
|
|
|
#endif
|
2002-11-07 15:50:30 +00:00
|
|
|
|
|
|
|
build_custom_used=0;
|
|
|
|
ubuild_custom_used=0;
|
2002-11-09 13:51:40 +00:00
|
|
|
|
|
|
|
res_editor=0;
|
2002-11-11 17:30:55 +00:00
|
|
|
|
|
|
|
build_last_page_define[0]=0;
|
|
|
|
ubuild_last_page_define[0]=0;
|
2003-03-06 21:24:19 +00:00
|
|
|
enable_last_page_cancel=0;
|
|
|
|
uenable_last_page_cancel=0;
|
2002-12-24 20:35:26 +00:00
|
|
|
|
2003-03-26 17:47:46 +00:00
|
|
|
next_used=false;
|
|
|
|
install_used=false;
|
|
|
|
comppage_used=false;
|
|
|
|
license_force_radio_used=false;
|
2003-04-21 13:32:34 +00:00
|
|
|
register_used=false;
|
|
|
|
unregister_used=false;
|
2003-03-26 17:47:46 +00:00
|
|
|
|
2002-12-24 20:35:26 +00:00
|
|
|
notify_hwnd=0;
|
2003-04-21 13:32:34 +00:00
|
|
|
|
|
|
|
uDefCodePage=CP_ACP;
|
2003-05-24 13:50:24 +00:00
|
|
|
|
|
|
|
use_first_insttype=true;
|
2003-05-25 17:51:20 +00:00
|
|
|
|
|
|
|
build_header.license_bg=-COLOR_BTNFACE;
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int CEXEBuild::getcurdbsize() { return cur_datablock->getlen(); }
|
|
|
|
|
|
|
|
int CEXEBuild::add_string(const char *string) // returns offset in stringblock
|
|
|
|
{
|
|
|
|
if (uninstall_mode) return add_string_uninst(string,1);
|
|
|
|
return add_string_main(string,1);
|
|
|
|
}
|
|
|
|
|
2003-05-24 13:50:24 +00:00
|
|
|
int CEXEBuild::add_intstring(const int i) // returns offset in stringblock
|
|
|
|
{
|
|
|
|
char i_str[64];
|
|
|
|
wsprintf(i_str, "%d", i);
|
|
|
|
return add_string(i_str);
|
|
|
|
}
|
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
// based on Dave Laundon's code
|
|
|
|
int CEXEBuild::preprocess_string(char *out, const char *in)
|
|
|
|
{
|
|
|
|
static const char VarNames[] =
|
|
|
|
"HWNDPARENT\0" // 0
|
|
|
|
"0\0" // 1
|
|
|
|
"1\0" // 2
|
|
|
|
"2\0" // 3
|
|
|
|
"3\0" // 4
|
|
|
|
"4\0" // 5
|
|
|
|
"5\0" // 6
|
|
|
|
"6\0" // 7
|
|
|
|
"7\0" // 8
|
|
|
|
"8\0" // 9
|
|
|
|
"9\0" // 10
|
|
|
|
"R0\0" // 11
|
|
|
|
"R1\0" // 12
|
|
|
|
"R2\0" // 13
|
|
|
|
"R3\0" // 14
|
|
|
|
"R4\0" // 15
|
|
|
|
"R5\0" // 16
|
|
|
|
"R6\0" // 17
|
|
|
|
"R7\0" // 18
|
|
|
|
"R8\0" // 19
|
|
|
|
"R9\0" // 20
|
2002-08-22 22:09:09 +00:00
|
|
|
"CMDLINE\0" // 21 everything before here doesn't have trailing slash removal
|
2002-08-21 16:23:29 +00:00
|
|
|
|
2002-08-22 22:09:09 +00:00
|
|
|
"INSTDIR\0" // 22
|
|
|
|
"OUTDIR\0" // 23
|
|
|
|
"EXEDIR\0" // 24
|
|
|
|
"LANGUAGE\0" // 25
|
2003-05-24 13:50:24 +00:00
|
|
|
"PLUGINSDIR\0" // 26
|
2003-05-25 13:12:07 +00:00
|
|
|
"PROGRAMFILES\0" // 27
|
|
|
|
"SMPROGRAMS\0" // 28
|
|
|
|
"SMSTARTUP\0" // 29
|
|
|
|
"DESKTOP\0" // 30
|
|
|
|
"STARTMENU\0" // 31
|
|
|
|
"QUICKLAUNCH\0" // 32
|
|
|
|
"TEMP\0" // 33
|
|
|
|
"WINDIR\0" // 34
|
|
|
|
"SYSDIR\0" // 35
|
2002-08-02 10:01:35 +00:00
|
|
|
;
|
|
|
|
|
|
|
|
const char *p=in;
|
2002-08-31 23:44:58 +00:00
|
|
|
while (*p)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
const char *np=CharNext(p);
|
|
|
|
|
|
|
|
if (np-p > 1) // multibyte char
|
|
|
|
{
|
|
|
|
int l=np-p;
|
|
|
|
while (l--)
|
|
|
|
{
|
|
|
|
int i = (unsigned char)*p++;
|
|
|
|
if (i >= VAR_CODES_START) {
|
|
|
|
*out++ = (char)255;
|
|
|
|
}
|
|
|
|
*out++=i;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
int i = (unsigned char)*p;
|
|
|
|
|
|
|
|
p=np;
|
|
|
|
|
|
|
|
// Test for characters extending into the variable codes
|
|
|
|
if (i >= VAR_CODES_START) {
|
|
|
|
*out++ = (char)255;
|
|
|
|
}
|
2002-08-31 23:44:58 +00:00
|
|
|
else if (i == '$')
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
2002-08-31 23:44:58 +00:00
|
|
|
if (*p == '$')
|
2002-08-02 10:01:35 +00:00
|
|
|
p++; // Can simply convert $$ to $ now
|
2002-08-31 23:44:58 +00:00
|
|
|
else
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
const char *pVarName;
|
|
|
|
for (
|
|
|
|
pVarName = VarNames, i = VAR_CODES_START;
|
|
|
|
strncmp(pVarName, p, strlen(pVarName));
|
|
|
|
pVarName += strlen(pVarName) + 1, i++
|
|
|
|
);
|
|
|
|
// Found?
|
2003-05-24 13:50:24 +00:00
|
|
|
if (*pVarName
|
|
|
|
#ifndef NSIS_CONFIG_PLUGIN_SUPPORT
|
|
|
|
&& i != VAR_CODES_START + 26
|
|
|
|
#endif
|
|
|
|
)
|
|
|
|
{
|
|
|
|
p += strlen(pVarName);
|
|
|
|
}
|
2002-08-02 10:01:35 +00:00
|
|
|
else // warning should go here
|
|
|
|
{
|
|
|
|
char tbuf[64];
|
|
|
|
strncpy(tbuf,p,63);
|
|
|
|
tbuf[63]=0;
|
|
|
|
if (strstr(tbuf," ")) strstr(tbuf," ")[0]=0;
|
2002-12-20 15:12:23 +00:00
|
|
|
warning("unknown variable \"%s\" detected, ignoring",tbuf);
|
2002-08-02 10:01:35 +00:00
|
|
|
i = '$';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*out++=i;
|
|
|
|
}
|
|
|
|
*out=0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CEXEBuild::add_string_main(const char *string, int process) // returns offset (in string block)
|
|
|
|
{
|
2002-09-29 20:25:15 +00:00
|
|
|
if (!*string) return 0;
|
2002-08-02 10:01:35 +00:00
|
|
|
|
2002-09-29 20:25:15 +00:00
|
|
|
if (*string == '$' && *(string+1) == '(') {
|
2002-09-29 21:32:35 +00:00
|
|
|
int idx = -1;
|
2002-09-29 20:25:15 +00:00
|
|
|
char *cp = strdup(string+2);
|
2002-09-29 21:32:35 +00:00
|
|
|
char *p = strchr(cp, ')');
|
2002-09-29 21:33:40 +00:00
|
|
|
if (p) {
|
2002-09-29 21:32:35 +00:00
|
|
|
*p = 0;
|
2002-12-20 15:12:23 +00:00
|
|
|
if (!strnicmp(cp,"un.",3)) {
|
|
|
|
warning("Installer language strings can't start with un. (%s)!", string);
|
|
|
|
free(cp);
|
|
|
|
return 0;
|
|
|
|
}
|
2002-12-20 19:14:57 +00:00
|
|
|
idx = GetUserString(cp);
|
2002-09-29 21:32:35 +00:00
|
|
|
}
|
2002-09-29 20:25:15 +00:00
|
|
|
free(cp);
|
2002-12-20 19:14:57 +00:00
|
|
|
if (idx >= 0) return -((int)(idx+1+(sizeof(common_strings)+sizeof(installer_strings))/sizeof(int)));
|
2002-09-29 20:25:15 +00:00
|
|
|
}
|
|
|
|
|
2002-10-01 17:16:49 +00:00
|
|
|
if (!process) return build_strlist.add(string,2);
|
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
char buf[4096];
|
|
|
|
preprocess_string(buf,string);
|
|
|
|
return build_strlist.add(buf,2);
|
|
|
|
}
|
|
|
|
|
|
|
|
int CEXEBuild::add_string_uninst(const char *string, int process) // returns offset (in string block)
|
|
|
|
{
|
2002-09-29 20:25:15 +00:00
|
|
|
if (!*string) return 0;
|
2002-08-02 10:01:35 +00:00
|
|
|
|
2002-09-29 20:25:15 +00:00
|
|
|
if (*string == '$' && *(string+1) == '(') {
|
2002-09-29 21:32:35 +00:00
|
|
|
int idx = -1;
|
2002-09-29 20:25:15 +00:00
|
|
|
char *cp = strdup(string+2);
|
2002-09-29 21:32:35 +00:00
|
|
|
char *p = strchr(cp, ')');
|
2002-09-29 21:33:40 +00:00
|
|
|
if (p) {
|
2002-09-29 21:32:35 +00:00
|
|
|
*p = 0;
|
2002-12-20 15:12:23 +00:00
|
|
|
if (strnicmp(cp,"un.",3)) {
|
|
|
|
warning("Uninstaller language strings must start with un. (%s)!", string);
|
|
|
|
free(cp);
|
|
|
|
return 0;
|
|
|
|
}
|
2002-12-20 19:14:57 +00:00
|
|
|
idx = GetUserString(cp);
|
2002-09-29 21:32:35 +00:00
|
|
|
}
|
2002-09-29 20:25:15 +00:00
|
|
|
free(cp);
|
2002-12-20 19:14:57 +00:00
|
|
|
if (idx >= 0) return -((int)(idx+1+(sizeof(common_strings)+sizeof(uninstall_strings))/sizeof(int)));
|
2002-09-29 20:25:15 +00:00
|
|
|
}
|
|
|
|
|
2002-10-01 17:16:49 +00:00
|
|
|
if (!process) return ubuild_strlist.add(string,2);
|
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
char buf[4096];
|
|
|
|
preprocess_string(buf,string);
|
|
|
|
return ubuild_strlist.add(buf,2);
|
|
|
|
}
|
|
|
|
|
2002-08-31 23:44:58 +00:00
|
|
|
// what it does is, when you pass it the offset of the last item added, it will determine if
|
2002-08-02 10:01:35 +00:00
|
|
|
// that data is already present in the datablock, and if so, reference it instead (and shorten
|
|
|
|
// the datablock as necessary). Reduces overhead if you want to add files to a couple places.
|
|
|
|
// Woo, an optimizing installer generator, now we're styling.
|
|
|
|
|
|
|
|
int CEXEBuild::datablock_optimize(int start_offset)
|
|
|
|
{
|
|
|
|
int this_len = cur_datablock->getlen()-start_offset;
|
|
|
|
int pos=0;
|
|
|
|
|
|
|
|
if (!build_optimize_datablock) return start_offset;
|
|
|
|
|
|
|
|
char *db=(char*)cur_datablock->get();
|
|
|
|
int first_int=*(int*)(db+start_offset);
|
|
|
|
if (this_len >= 4) while (pos < start_offset)
|
|
|
|
{
|
|
|
|
int this_int = *(int*)(db+pos);
|
|
|
|
if (this_int == first_int && !memcmp(db+pos,db+start_offset,this_len))
|
|
|
|
{
|
|
|
|
db_opt_save+=this_len;
|
|
|
|
cur_datablock->resize(max(start_offset,pos+this_len));
|
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
pos += 4 + (this_int&0x7fffffff);
|
|
|
|
}
|
|
|
|
|
|
|
|
return start_offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CEXEBuild::add_data(const char *data, int length, IGrowBuf *dblock) // returns offset
|
|
|
|
{
|
|
|
|
build_compressor_set=true;
|
|
|
|
|
|
|
|
int done=0;
|
|
|
|
|
|
|
|
if (length < 0)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: add_data() called with length=%d\n",length);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!dblock) dblock=cur_datablock;
|
|
|
|
|
|
|
|
int st=dblock->getlen();
|
|
|
|
|
|
|
|
#ifdef NSIS_CONFIG_COMPRESSION_SUPPORT
|
|
|
|
if (!build_compress_whole && build_compress)
|
|
|
|
{
|
|
|
|
// grow datablock so that there is room to compress into
|
|
|
|
int bufferlen=length+1024+length/4; // give a nice 25% extra space
|
|
|
|
dblock->resize(st+bufferlen+sizeof(int));
|
|
|
|
|
|
|
|
int n;
|
2002-08-31 23:44:58 +00:00
|
|
|
if ((n=compressor->Init(9)) != C_OK)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
ERROR_MSG("Internal compiler error #12345: deflateInit() failed(%d).\n",n);
|
|
|
|
extern void quit(); quit();
|
|
|
|
}
|
|
|
|
|
|
|
|
compressor->SetNextIn((char*)data, length);
|
|
|
|
compressor->SetNextOut((char*)dblock->get() + st + sizeof(int), bufferlen);
|
|
|
|
|
|
|
|
compressor->Compress(C_FINISH);
|
|
|
|
|
|
|
|
int used=bufferlen-compressor->GetAvailOut();
|
|
|
|
|
|
|
|
// never store compressed if output buffer is full
|
|
|
|
if (compressor->GetAvailOut() && (build_compress == 2 || used < length))
|
|
|
|
{
|
|
|
|
done=1;
|
|
|
|
dblock->resize(st+used+sizeof(int));
|
|
|
|
|
|
|
|
*((int*)((char *)dblock->get()+st)) = used|0x80000000;
|
|
|
|
if (dblock == cur_datablock)
|
|
|
|
{
|
|
|
|
int nst=datablock_optimize(st);
|
|
|
|
if (nst == st) db_comp_save+=length-used;
|
|
|
|
else st=nst;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
compressor->End();
|
|
|
|
}
|
|
|
|
#endif // NSIS_CONFIG_COMPRESSION_SUPPORT
|
|
|
|
|
|
|
|
if (!done)
|
|
|
|
{
|
|
|
|
dblock->resize(st);
|
|
|
|
dblock->add(&length,sizeof(int));
|
|
|
|
dblock->add(data,length);
|
|
|
|
if (dblock == cur_datablock)
|
|
|
|
{
|
|
|
|
st=datablock_optimize(st);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dblock == cur_datablock)
|
|
|
|
{
|
|
|
|
db_full_size += length + sizeof(int);
|
|
|
|
}
|
|
|
|
|
|
|
|
return st;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CEXEBuild::add_label(const char *name)
|
|
|
|
{
|
|
|
|
if (!build_cursection && !uninstall_mode)
|
|
|
|
{
|
2002-08-31 23:44:58 +00:00
|
|
|
ERROR_MSG("Error: Label declaration not valid outside of function/section\n");
|
2002-08-02 10:01:35 +00:00
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
if ((name[0] >= '0' && name[0] <= '9') || name[0] == '-' || name[0] == ' ' || name[0] == ':')
|
|
|
|
{
|
2002-08-31 23:44:58 +00:00
|
|
|
ERROR_MSG("Error: labels must not begin with 0-9, -, :, or a space.\n");
|
2002-08-02 10:01:35 +00:00
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
int cs;
|
|
|
|
int ce;
|
|
|
|
if (build_cursection)
|
|
|
|
{
|
|
|
|
cs=build_cursection->code;
|
|
|
|
ce=cs+build_cursection->code_size;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cs=build_uninst.code;
|
|
|
|
ce=cs+build_uninst.code_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *p=strdup(name);
|
|
|
|
if (p[strlen(p)-1] == ':') p[strlen(p)-1]=0;
|
|
|
|
int offs=ns_label.add(p,0);
|
|
|
|
free(p);
|
|
|
|
|
|
|
|
int n=cur_labels->getlen()/sizeof(section);
|
|
|
|
if (n)
|
|
|
|
{
|
|
|
|
section *t=(section*)cur_labels->get();
|
|
|
|
while (n--)
|
|
|
|
{
|
2002-08-31 23:44:58 +00:00
|
|
|
if ((*name == '.' || (t->code >= cs && t->code <= ce)) &&
|
2002-08-02 10:01:35 +00:00
|
|
|
t->name_ptr==offs)
|
|
|
|
{
|
|
|
|
if (*name == '.') ERROR_MSG("Error: global label \"%s\" already declared\n",name);
|
|
|
|
else ERROR_MSG("Error: label \"%s\" already declared in section/function\n",name);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
t++;
|
|
|
|
}
|
|
|
|
}
|
2002-08-31 23:44:58 +00:00
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
section s={0,};
|
|
|
|
s.name_ptr = offs;
|
|
|
|
s.code = ce;
|
|
|
|
cur_labels->add(&s,sizeof(s));
|
|
|
|
|
|
|
|
return PS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CEXEBuild::add_function(const char *funname)
|
|
|
|
{
|
|
|
|
if (build_cursection_isfunc)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: Function open when creating function (use FunctionEnd first)\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
if (build_cursection)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: Section open when creating function (use SectionEnd first)\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
if (!funname[0])
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: Function must have a name\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strnicmp(funname,"un.",3))
|
|
|
|
{
|
|
|
|
set_uninstall_mode(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
int addr=ns_func.add(funname,0);
|
|
|
|
int x;
|
|
|
|
int n=cur_functions->getlen()/sizeof(section);
|
|
|
|
section *tmp=(section*)cur_functions->get();
|
|
|
|
for (x = 0; x < n; x ++)
|
|
|
|
{
|
|
|
|
if (tmp[x].name_ptr == addr)
|
2003-04-17 15:27:12 +00:00
|
|
|
{
|
|
|
|
ERROR_MSG("Error: Function named \"%s\" already exists.\n",funname);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
cur_functions->resize((n+1)*sizeof(section));
|
|
|
|
build_cursection=((section*)cur_functions->get())+n;
|
|
|
|
build_cursection_isfunc=1;
|
|
|
|
build_cursection->name_ptr=addr;
|
|
|
|
build_cursection->code=cur_entries->getlen()/sizeof(entry);
|
|
|
|
build_cursection->code_size=0;
|
2002-10-02 15:01:06 +00:00
|
|
|
build_cursection->install_types=0;
|
|
|
|
build_cursection->flags=0;
|
2002-08-02 10:01:35 +00:00
|
|
|
build_cursection->size_kb=0;
|
|
|
|
return PS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CEXEBuild::function_end()
|
|
|
|
{
|
|
|
|
if (!build_cursection_isfunc)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: No function open, FunctionEnd called\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
build_cursection_isfunc=0;
|
|
|
|
build_cursection=NULL;
|
|
|
|
|
|
|
|
// add ret.
|
|
|
|
entry ent={EW_RET,};
|
|
|
|
cur_entries->add(&ent,sizeof(entry));
|
|
|
|
if (!uninstall_mode) build_header.common.num_entries++;
|
|
|
|
else build_uninst.common.num_entries++;
|
|
|
|
|
|
|
|
set_uninstall_mode(0);
|
|
|
|
return PS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int CEXEBuild::section_add_flags(int flags)
|
|
|
|
{
|
|
|
|
if (uninstall_mode)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: can't modify flags of uninstall section\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
if (!build_cursection || build_cursection_isfunc)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: can't modify flags when no section is open\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
2002-10-02 15:01:06 +00:00
|
|
|
build_cursection->flags|=flags;
|
|
|
|
return PS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CEXEBuild::section_add_install_type(int inst_type)
|
|
|
|
{
|
|
|
|
if (uninstall_mode)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: can't modify flags of uninstall section\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
if (!build_cursection || build_cursection_isfunc)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: can't modify flags when no section is open\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
2002-10-19 12:02:18 +00:00
|
|
|
if (build_cursection->install_types == ~(int)0)
|
|
|
|
build_cursection->install_types = 0;
|
2002-10-02 15:01:06 +00:00
|
|
|
build_cursection->install_types|=inst_type;
|
2002-08-02 10:01:35 +00:00
|
|
|
return PS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CEXEBuild::section_add_size_kb(int kb)
|
|
|
|
{
|
|
|
|
if (build_cursection)
|
|
|
|
{
|
|
|
|
build_cursection->size_kb+=kb;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int CEXEBuild::section_end()
|
|
|
|
{
|
|
|
|
if (build_cursection_isfunc)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: SectionEnd specified in function (not section)\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
else if (uninstall_mode)
|
|
|
|
{
|
|
|
|
entry ent={EW_RET,};
|
|
|
|
cur_entries->add(&ent,sizeof(entry));
|
|
|
|
build_uninst.common.num_entries++;
|
|
|
|
set_uninstall_mode(0);
|
|
|
|
}
|
|
|
|
else if (!build_cursection)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: SectionEnd specified and no sections open\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
2002-08-31 23:44:58 +00:00
|
|
|
else
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
entry ent={EW_RET,};
|
|
|
|
cur_entries->add(&ent,sizeof(entry));
|
|
|
|
build_header.common.num_entries++;
|
|
|
|
}
|
|
|
|
build_cursection=NULL;
|
|
|
|
return PS_OK;
|
|
|
|
}
|
|
|
|
|
2003-05-24 13:50:24 +00:00
|
|
|
int CEXEBuild::add_section(const char *secname, const char *file, int line, const char *defname, int expand/*=0*/)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
if (build_cursection_isfunc)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: Section can't create section (already in function, use FunctionEnd first)\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
if (build_cursection || uninstall_mode)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: Section already open, call SectionEnd first\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
if (!stricmp(secname,"uninstall"))
|
|
|
|
{
|
|
|
|
if (defname[0])
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: Uninstall section cannot have index output define\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
if (build_uninst.code_size >= 0)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: Uninstall section already specified\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
if (subsection_open_cnt)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: Uninstall section specified inside SubSection\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
build_uninst.code=ubuild_entries.getlen()/sizeof(entry);
|
|
|
|
build_uninst.code_size=0;
|
|
|
|
set_uninstall_mode(1);
|
|
|
|
build_cursection=NULL;
|
|
|
|
return PS_OK;
|
|
|
|
}
|
|
|
|
|
2002-10-02 21:31:35 +00:00
|
|
|
char *name = (char*)secname;
|
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
int n=(build_sections.getlen()/sizeof(section));
|
|
|
|
build_sections.resize(build_sections.getlen()+sizeof(section));
|
|
|
|
build_cursection=((section*)build_sections.get()) + n;
|
2002-10-02 15:01:06 +00:00
|
|
|
build_cursection->flags=SF_SELECTED;
|
2002-10-02 21:31:35 +00:00
|
|
|
build_cursection->flags|=expand?SF_EXPAND:0;
|
|
|
|
if (secname[0]=='!' || (secname[0]=='-' && secname[1]=='!')) {
|
|
|
|
name++;
|
|
|
|
build_cursection->flags|=SF_BOLD;
|
|
|
|
}
|
|
|
|
build_cursection->flags|=expand?SF_EXPAND:0;
|
|
|
|
build_cursection->name_ptr=add_string(secname[0]=='-'&&secname[1]?++name:name);
|
2002-08-02 10:01:35 +00:00
|
|
|
build_cursection->code=cur_entries->getlen()/sizeof(entry);
|
|
|
|
build_cursection->code_size=0;
|
|
|
|
build_cursection->size_kb=0;
|
2002-10-19 12:02:18 +00:00
|
|
|
build_cursection->install_types=*name?0:~(int)0;
|
2002-08-02 10:01:35 +00:00
|
|
|
|
|
|
|
if (secname[0]=='-')
|
|
|
|
{
|
2002-10-02 15:01:06 +00:00
|
|
|
build_cursection->flags|=secname[1]?SF_SUBSEC:SF_SUBSECEND;
|
2002-08-02 10:01:35 +00:00
|
|
|
build_cursection=NULL;
|
2002-08-31 23:44:58 +00:00
|
|
|
entry ent={EW_RET,};
|
|
|
|
cur_entries->add(&ent,sizeof(entry));
|
|
|
|
build_header.common.num_entries++;
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (defname[0])
|
|
|
|
{
|
|
|
|
char buf[32];
|
|
|
|
wsprintf(buf,"%d",build_header.num_sections);
|
|
|
|
if (definedlist.add(defname,buf))
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: \"%s\" already defined, can't assign section index!\n",defname);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
}
|
2002-08-31 23:44:58 +00:00
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
build_header.num_sections++;
|
|
|
|
|
|
|
|
return PS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CEXEBuild::make_sure_not_in_secorfunc(const char *str)
|
|
|
|
{
|
|
|
|
if (build_cursection || uninstall_mode)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: command %s not valid in %s\n",str,build_cursection_isfunc?"function":"section");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
return PS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CEXEBuild::add_entry(const entry *ent)
|
|
|
|
{
|
2002-08-31 23:44:58 +00:00
|
|
|
if (!build_cursection && !uninstall_mode)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
ERROR_MSG("Error: Can't add entry, no section or function is open!\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
cur_entries->add(ent,sizeof(entry));
|
|
|
|
|
|
|
|
if (!uninstall_mode)
|
|
|
|
{
|
|
|
|
if (!build_cursection_isfunc && build_cursection->name_ptr >=0 && build_strlist.get()[build_cursection->name_ptr] == '-')
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: cannot add entry to divider section\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
build_cursection->code_size++;
|
|
|
|
build_header.common.num_entries++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
build_uninst.common.num_entries++;
|
|
|
|
if (build_cursection) build_cursection->code_size++;
|
|
|
|
else build_uninst.code_size++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return PS_OK;
|
|
|
|
}
|
|
|
|
|
2003-04-02 15:49:13 +00:00
|
|
|
int CEXEBuild::add_entry_direct(int which, int o0, int o1, int o2, int o3, int o4, int o5 /*o#=0*/)
|
|
|
|
{
|
|
|
|
entry ent;
|
|
|
|
ent.which = which;
|
|
|
|
ent.offsets[0] = o0;
|
|
|
|
ent.offsets[1] = o1;
|
|
|
|
ent.offsets[2] = o2;
|
|
|
|
ent.offsets[3] = o3;
|
|
|
|
ent.offsets[4] = o4;
|
|
|
|
ent.offsets[5] = o5;
|
|
|
|
return add_entry(&ent);
|
|
|
|
}
|
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
int CEXEBuild::resolve_jump_int(const char *fn, int *a, int offs, int start, int end)
|
|
|
|
{
|
2002-08-31 23:44:58 +00:00
|
|
|
if (*a > 0)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
char *lname=(char*)ns_label.get()+*a;
|
|
|
|
if (lname[0] == '-' || lname[0]=='+')
|
|
|
|
{
|
|
|
|
*a=offs+atoi(lname)+1;
|
|
|
|
}
|
2002-08-31 23:44:58 +00:00
|
|
|
else
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
section *s = (section*)cur_labels->get();
|
|
|
|
int n=cur_labels->getlen()/sizeof(section);
|
|
|
|
while (n-->0)
|
|
|
|
{
|
|
|
|
if ((*lname == '.' || (s->code >= start && s->code <= end)) && s->name_ptr == *a)
|
|
|
|
{
|
|
|
|
*a = s->code+1; // jumps are to the absolute position, +1 (to differentiate between no jump, and jumping to offset 0)
|
2002-10-02 15:01:06 +00:00
|
|
|
s->flags++;
|
2002-08-02 10:01:35 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
|
|
|
|
ERROR_MSG("Error: could not resolve label \"%s\" in %s\n",lname,fn);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (*a < 0) // to jump to a user variable target, -variable_index-1 is already stored.
|
|
|
|
{
|
|
|
|
}
|
|
|
|
// otherwise, *a is 0, which means no jump and we also leave it intact
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CEXEBuild::resolve_call_int(const char *fn, const char *str, int fptr, int *ofs)
|
|
|
|
{
|
|
|
|
if (fptr < 0) return 0;
|
|
|
|
int nf=cur_functions->getlen()/sizeof(section);
|
|
|
|
section *sec=(section *)cur_functions->get();
|
|
|
|
while (nf-- > 0)
|
|
|
|
{
|
|
|
|
if (sec->name_ptr>0 && sec->name_ptr == fptr)
|
|
|
|
{
|
|
|
|
ofs[0]=sec->code;
|
2002-10-02 15:01:06 +00:00
|
|
|
sec->flags++;
|
2002-08-02 10:01:35 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
sec++;
|
|
|
|
}
|
|
|
|
ERROR_MSG("Error: resolving %s function \"%s\" in %s\n",str,(char*)ns_func.get()+fptr,fn);
|
2003-04-17 15:27:12 +00:00
|
|
|
ERROR_MSG("Note: uninstall functions must begin with \"un.\", and install functions must not\n");
|
|
|
|
return 1;
|
|
|
|
}
|
2002-08-02 10:01:35 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int CEXEBuild::resolve_instruction(const char *fn, const char *str, entry *w, int offs, int start, int end)
|
|
|
|
{
|
|
|
|
if (w->which == EW_NOP)
|
|
|
|
{
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[0],offs,start,end)) return 1;
|
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
#ifdef NSIS_SUPPORT_MESSAGEBOX
|
2002-08-02 10:01:35 +00:00
|
|
|
else if (w->which == EW_MESSAGEBOX)
|
|
|
|
{
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[3],offs,start,end)) return 1;
|
2003-03-29 17:16:09 +00:00
|
|
|
if (resolve_jump_int(fn,&w->offsets[5],offs,start,end)) return 1;
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
else if (w->which == EW_IFFILEEXISTS)
|
|
|
|
{
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[1],offs,start,end)) return 1;
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[2],offs,start,end)) return 1;
|
|
|
|
}
|
2003-04-02 19:54:53 +00:00
|
|
|
else if (w->which == EW_IFFLAG)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[0],offs,start,end)) return 1;
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[1],offs,start,end)) return 1;
|
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
#ifdef NSIS_SUPPORT_STROPTS
|
2002-08-02 10:01:35 +00:00
|
|
|
else if (w->which == EW_STRCMP)
|
|
|
|
{
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[2],offs,start,end)) return 1;
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[3],offs,start,end)) return 1;
|
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
#endif
|
|
|
|
#ifdef NSIS_SUPPORT_INTOPTS
|
2003-03-20 20:49:13 +00:00
|
|
|
else if (w->which == EW_INTCMP)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[2],offs,start,end)) return 1;
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[3],offs,start,end)) return 1;
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[4],offs,start,end)) return 1;
|
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
#endif
|
|
|
|
#ifdef NSIS_SUPPORT_HWNDS
|
2002-08-02 10:01:35 +00:00
|
|
|
else if (w->which == EW_ISWINDOW)
|
|
|
|
{
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[1],offs,start,end)) return 1;
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[2],offs,start,end)) return 1;
|
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
else if (w->which == EW_CALL)
|
|
|
|
{
|
|
|
|
if (w->offsets[0] >= 0 && w->offsets[1]) // get as jump
|
|
|
|
{
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[0],offs,start,end)) return 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (w->offsets[0] >= 0 && resolve_call_int(fn,str,w->offsets[0],w->offsets)) return 1;
|
|
|
|
// if w->offsets[0] >= 0, EW_CALL requires that it 1-based.
|
2002-08-31 23:44:58 +00:00
|
|
|
// otherwise, if < 0, it needs an increment anyway (since it
|
2002-08-02 10:01:35 +00:00
|
|
|
// was encoded with a -2 base, to prevent it looking like an
|
|
|
|
// empty string "")
|
2002-08-31 23:44:58 +00:00
|
|
|
w->offsets[0]++;
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
#ifdef NSIS_SUPPORT_STROPTS
|
2002-08-02 10:01:35 +00:00
|
|
|
else if (w->which == EW_GETFUNCTIONADDR)
|
|
|
|
{
|
|
|
|
char buf[32];
|
|
|
|
if (w->offsets[1] < 0)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: GetFunctionAddress requires a real function to get address of.\n");
|
|
|
|
return 1;
|
|
|
|
}
|
2002-08-31 23:44:58 +00:00
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
if (resolve_call_int(fn,str,w->offsets[1],&w->offsets[1])) return 1;
|
|
|
|
|
|
|
|
w->which=EW_ASSIGNVAR;
|
|
|
|
wsprintf(buf,"%d",w->offsets[1]+1); // +1 here to make 1-based.
|
|
|
|
w->offsets[1]=add_string(buf);
|
|
|
|
}
|
|
|
|
else if (w->which == EW_GETLABELADDR)
|
|
|
|
{
|
|
|
|
char buf[32];
|
|
|
|
if (resolve_jump_int(fn,&w->offsets[1],offs,start,end)) return 1;
|
|
|
|
w->which=EW_ASSIGNVAR;
|
|
|
|
wsprintf(buf,"%d",w->offsets[1]);
|
|
|
|
w->offsets[1]=add_string(buf);
|
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CEXEBuild::resolve_coderefs(const char *str)
|
|
|
|
{
|
|
|
|
// resolve jumps&calls
|
|
|
|
{
|
2002-08-31 23:44:58 +00:00
|
|
|
section *sec=(section *)cur_functions->get();
|
2002-08-02 10:01:35 +00:00
|
|
|
int l=cur_functions->getlen()/sizeof(section);
|
2002-11-01 20:34:55 +00:00
|
|
|
entry *w=(entry *)cur_entries->get();
|
2002-08-02 10:01:35 +00:00
|
|
|
while (l-- > 0)
|
|
|
|
{
|
|
|
|
int x;
|
|
|
|
for (x = sec->code; x < sec->code+sec->code_size; x ++)
|
|
|
|
{
|
|
|
|
char fname[1024];
|
|
|
|
wsprintf(fname,"function \"%s\"",ns_func.get()+sec->name_ptr);
|
|
|
|
if (resolve_instruction(fname,str,w+x,x,sec->code,sec->code+sec->code_size)) return 1;
|
|
|
|
}
|
|
|
|
sec++;
|
|
|
|
}
|
|
|
|
if (uninstall_mode)
|
|
|
|
{
|
|
|
|
int x;
|
|
|
|
for (x = build_uninst.code; x < build_uninst.code+build_uninst.code_size; x ++)
|
|
|
|
if (resolve_instruction("\"uninstall section\"",str,w+x,x,build_uninst.code,build_uninst.code+build_uninst.code_size)) return 1;
|
2002-11-01 20:34:55 +00:00
|
|
|
|
|
|
|
#ifdef NSIS_SUPPORT_CODECALLBACKS
|
|
|
|
if (ubuild_pages.getlen()) {
|
|
|
|
page *p=(page *)ubuild_pages.get();
|
|
|
|
int i = 0;
|
|
|
|
while (i < build_uninst.common.num_pages) {
|
2003-03-18 15:45:25 +00:00
|
|
|
if (resolve_call_int("uninstall pages","pre-page",p->prefunc,&p->prefunc)) return 1;
|
2003-03-24 14:50:35 +00:00
|
|
|
if (resolve_call_int("uninstall pages",p->id?"show-page":"leave-page",p->showfunc,&p->showfunc)) return 1;
|
2003-03-18 15:45:25 +00:00
|
|
|
if (resolve_call_int("uninstall pages","leave-page",p->leavefunc,&p->leavefunc)) return 1;
|
2002-11-01 20:34:55 +00:00
|
|
|
p++;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int cnt=0;
|
|
|
|
sec=(section *)build_sections.get();
|
|
|
|
l=build_sections.getlen()/sizeof(section);
|
|
|
|
while (l-- > 0)
|
|
|
|
{
|
|
|
|
int x;
|
|
|
|
for (x = sec->code; x < sec->code+sec->code_size; x ++)
|
|
|
|
{
|
|
|
|
char fname[1024];
|
2002-09-29 20:25:15 +00:00
|
|
|
if (sec->name_ptr) wsprintf(fname,"section \"%s\" (%d)",build_strlist.get()+sec->name_ptr,cnt);
|
2002-08-02 10:01:35 +00:00
|
|
|
else wsprintf(fname,"unnamed section (%d)",cnt);
|
|
|
|
if (resolve_instruction(fname,str,w+x,x,sec->code,sec->code+sec->code_size)) return 1;
|
|
|
|
}
|
|
|
|
sec++;
|
|
|
|
cnt++;
|
|
|
|
}
|
2002-11-01 20:34:55 +00:00
|
|
|
#ifdef NSIS_SUPPORT_CODECALLBACKS
|
|
|
|
if (build_pages.getlen()) {
|
|
|
|
page *p=(page *)build_pages.get();
|
|
|
|
int i = 0;
|
|
|
|
while (i < build_header.common.num_pages) {
|
2003-03-24 14:50:35 +00:00
|
|
|
if (resolve_call_int("pages","pre-page",p->prefunc,&p->prefunc)) return 1;
|
|
|
|
if (resolve_call_int("pages",p->id?"show-page":"leave-page",p->showfunc,&p->showfunc)) return 1;
|
|
|
|
if (resolve_call_int("pages","leave-page",p->leavefunc,&p->leavefunc)) return 1;
|
2002-11-01 20:34:55 +00:00
|
|
|
p++;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// optimize unused functions
|
|
|
|
{
|
2002-08-31 23:44:58 +00:00
|
|
|
section *sec=(section *)cur_functions->get();
|
2002-08-02 10:01:35 +00:00
|
|
|
int l=cur_functions->getlen()/sizeof(section);
|
|
|
|
entry *w=(entry*)cur_entries->get();
|
|
|
|
while (l-- > 0)
|
|
|
|
{
|
|
|
|
if (sec->name_ptr)
|
|
|
|
{
|
2002-10-02 15:01:06 +00:00
|
|
|
if (!sec->flags)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
if (sec->code_size>0)
|
|
|
|
{
|
|
|
|
warning("%s function \"%s\" not referenced - zeroing code (%d-%d) out\n",str,
|
|
|
|
ns_func.get()+sec->name_ptr,
|
|
|
|
sec->code,sec->code+sec->code_size);
|
|
|
|
memset(w+sec->code,0,sec->code_size*sizeof(entry));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sec++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// give warnings on unused labels
|
|
|
|
{
|
|
|
|
section *t=(section*)cur_labels->get();
|
|
|
|
int n=cur_labels->getlen()/sizeof(section);
|
|
|
|
while (n-->0)
|
|
|
|
{
|
2002-10-02 15:01:06 +00:00
|
|
|
if (!t->flags)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
char *n=(char*)ns_label.get()+t->name_ptr;
|
|
|
|
if (*n == '.') warning("global label \"%s\" not used",n);
|
|
|
|
else warning("label \"%s\" not used",n);
|
|
|
|
}
|
|
|
|
t++;
|
|
|
|
}
|
|
|
|
}
|
2002-08-31 23:44:58 +00:00
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int CEXEBuild::write_output(void)
|
|
|
|
{
|
|
|
|
#ifndef NSIS_CONFIG_CRC_SUPPORT
|
|
|
|
build_crcchk=0;
|
|
|
|
#endif
|
|
|
|
if (has_called_write_output)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error (write_output): write_output already called, can't continue\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
has_called_write_output++;
|
|
|
|
if (!build_output_filename[0])
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: invalid script: never had OutFile command\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
int ns=build_sections.getlen()/sizeof(section);
|
|
|
|
if (!ns)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: invalid script: no sections specified\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
2002-09-29 20:25:15 +00:00
|
|
|
}
|
2002-08-02 10:01:35 +00:00
|
|
|
|
|
|
|
if (!build_entries.getlen())
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: invalid script: no entries specified\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (build_cursection || uninstall_mode)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: Section left open at EOF\n");
|
|
|
|
return PS_ERROR;
|
2002-08-31 23:44:58 +00:00
|
|
|
}
|
2002-08-02 10:01:35 +00:00
|
|
|
|
|
|
|
if (subsection_open_cnt)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: SubSection left open at EOF\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
// deal with functions, for both install and uninstall modes.
|
|
|
|
if (build_cursection_isfunc)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: Function still open at EOF, cannot proceed\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2002-11-25 16:50:05 +00:00
|
|
|
#ifdef NSIS_CONFIG_PLUGIN_SUPPORT
|
|
|
|
int err=add_plugins_dir_initializer();
|
|
|
|
if (err != PS_OK)
|
|
|
|
return err;
|
|
|
|
#endif //NSIS_CONFIG_PLUGIN_SUPPORT
|
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
#ifdef NSIS_CONFIG_UNINSTALL_SUPPORT
|
|
|
|
if (ubuild_entries.getlen())
|
|
|
|
{
|
|
|
|
if (!uninstaller_writes_used)
|
|
|
|
{
|
2002-08-21 16:23:29 +00:00
|
|
|
warning("Uninstall section found but WriteUninstaller never used - no uninstaller will be created.");
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2003-03-18 20:36:52 +00:00
|
|
|
build_uninst.common.flags|=build_header.common.flags&(CH_FLAGS_PROGRESS_COLORED|CH_FLAGS_NO_ROOT_DIR);
|
2002-08-31 23:44:58 +00:00
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
set_uninstall_mode(1);
|
|
|
|
#ifdef NSIS_SUPPORT_CODECALLBACKS
|
|
|
|
if (resolve_call_int("uninstall callback","un.callbacks",ns_func.find("un.onInit",0),&build_uninst.common.code_onInit)) return PS_ERROR;
|
|
|
|
if (resolve_call_int("uninstall callback","un.callbacks",ns_func.find("un.onUninstSuccess",0),&build_uninst.common.code_onInstSuccess)) return PS_ERROR;
|
|
|
|
if (resolve_call_int("uninstall callback","un.callbacks",ns_func.find("un.onUninstFailed",0),&build_uninst.common.code_onInstFailed)) return PS_ERROR;
|
|
|
|
if (resolve_call_int("uninstall callback","un.callbacks",ns_func.find("un.onUserAbort",0),&build_uninst.common.code_onUserAbort)) return PS_ERROR;
|
2002-09-18 18:39:24 +00:00
|
|
|
#ifdef NSIS_CONFIG_ENHANCEDUI_SUPPORT
|
2002-11-01 20:34:55 +00:00
|
|
|
if (resolve_call_int("uninstall callback","un.callbacks",ns_func.find("un.onGUIInit",0),&build_uninst.common.code_onGUIInit)) return PS_ERROR;
|
2003-05-09 21:11:14 +00:00
|
|
|
if (resolve_call_int("uninstall callback","un.callbacks",ns_func.find("un.onGUIEnd",0),&build_uninst.common.code_onGUIEnd)) return PS_ERROR;
|
2002-09-18 18:39:24 +00:00
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif//NSIS_SUPPORT_CODECALLBACKS
|
2002-11-01 20:34:55 +00:00
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
if (resolve_coderefs("uninstall")) return PS_ERROR;
|
|
|
|
set_uninstall_mode(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (uninstaller_writes_used)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: no Uninstall section specified, but WriteUninstaller used %d time(s)\n",uninstaller_writes_used);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef NSIS_SUPPORT_CODECALLBACKS
|
|
|
|
if (resolve_call_int("install callback",".callbacks",ns_func.find(".onInit",0),&build_header.common.code_onInit)) return PS_ERROR;
|
|
|
|
if (resolve_call_int("install callback",".callbacks",ns_func.find(".onInstSuccess",0),&build_header.common.code_onInstSuccess)) return PS_ERROR;
|
|
|
|
if (resolve_call_int("install callback",".callbacks",ns_func.find(".onInstFailed",0),&build_header.common.code_onInstFailed)) return PS_ERROR;
|
|
|
|
if (resolve_call_int("install callback",".callbacks",ns_func.find(".onUserAbort",0),&build_header.common.code_onUserAbort)) return PS_ERROR;
|
|
|
|
if (resolve_call_int("install callback",".callbacks",ns_func.find(".onVerifyInstDir",0),&build_header.code_onVerifyInstDir)) return PS_ERROR;
|
2002-09-18 18:39:24 +00:00
|
|
|
#ifdef NSIS_CONFIG_ENHANCEDUI_SUPPORT
|
2002-11-01 20:34:55 +00:00
|
|
|
if (resolve_call_int("install callback",".callbacks",ns_func.find(".onGUIInit",0),&build_header.common.code_onGUIInit)) return PS_ERROR;
|
2003-05-09 21:11:14 +00:00
|
|
|
if (resolve_call_int("install callback",".callbacks",ns_func.find(".onGUIEnd",0),&build_header.common.code_onGUIEnd)) return PS_ERROR;
|
2002-09-18 18:39:24 +00:00
|
|
|
if (resolve_call_int("install callback",".callbacks",ns_func.find(".onMouseOverSection",0),&build_header.code_onMouseOverSection)) return PS_ERROR;
|
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
#ifdef NSIS_CONFIG_COMPONENTPAGE
|
|
|
|
if (resolve_call_int("install callback",".callbacks",ns_func.find(".onSelChange",0),&build_header.code_onSelChange)) return PS_ERROR;
|
|
|
|
#endif//NSIS_CONFIG_COMPONENTPAGE
|
|
|
|
#endif//NSIS_SUPPORT_CODECALLBACKS
|
2002-08-31 23:44:58 +00:00
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
if (resolve_coderefs("install")) return PS_ERROR;
|
|
|
|
|
2003-05-24 13:50:24 +00:00
|
|
|
// set sections to the first insttype
|
|
|
|
if (use_first_insttype && build_header.install_types[0])
|
|
|
|
{
|
|
|
|
int n = build_sections.getlen()/sizeof(section);
|
|
|
|
section *sections = (section *) build_sections.get();
|
|
|
|
for (int i = 0; i < n; i++)
|
|
|
|
{
|
|
|
|
if ((sections[i].install_types & 1) == 0)
|
|
|
|
sections[i].flags &= ~SF_SELECTED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
#ifdef NSIS_CONFIG_VISIBLE_SUPPORT
|
2002-11-01 20:34:55 +00:00
|
|
|
{
|
2002-11-09 13:51:40 +00:00
|
|
|
SCRIPT_MSG("Processing pages... ");
|
2002-11-01 20:34:55 +00:00
|
|
|
page pg = {
|
|
|
|
0,
|
|
|
|
#ifdef NSIS_SUPPORT_CODECALLBACKS
|
|
|
|
-1,
|
2002-11-11 19:19:02 +00:00
|
|
|
-1,
|
2003-01-24 19:40:20 +00:00
|
|
|
-1,
|
2002-11-01 20:34:55 +00:00
|
|
|
#endif
|
2002-11-11 19:19:02 +00:00
|
|
|
0
|
2002-11-01 20:34:55 +00:00
|
|
|
};
|
|
|
|
int add_pages=!build_pages.getlen();
|
|
|
|
int add_uninst_pages=!ubuild_pages.getlen();
|
|
|
|
|
|
|
|
int license=0;
|
|
|
|
int selcom=0;
|
|
|
|
int dir=0;
|
|
|
|
int uninst=0;
|
|
|
|
int instlog=0;
|
|
|
|
int main=2;
|
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
#ifdef NSIS_CONFIG_SILENT_SUPPORT
|
2003-03-18 20:36:52 +00:00
|
|
|
if (!(build_header.common.flags&(CH_FLAGS_SILENT|CH_FLAGS_SILENT_LOG)))
|
2002-11-01 20:34:55 +00:00
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
2002-11-01 20:34:55 +00:00
|
|
|
#ifdef NSIS_CONFIG_LICENSEPAGE
|
|
|
|
if (!IsNotSet(installer.licensedata)) license++;
|
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
#ifdef NSIS_CONFIG_COMPONENTPAGE
|
2002-11-01 20:34:55 +00:00
|
|
|
if (!IsNotSet(installer.componenttext)) selcom++;
|
|
|
|
#endif
|
|
|
|
if (!IsNotSet(installer.text)) dir++;
|
|
|
|
|
|
|
|
if (!add_pages) {
|
|
|
|
int i=0;
|
|
|
|
page *p=(page *) build_pages.get();
|
|
|
|
while (i!=build_header.common.num_pages) {
|
|
|
|
switch (p->id) {
|
|
|
|
#ifdef NSIS_CONFIG_LICENSEPAGE
|
|
|
|
case NSIS_PAGE_LICENSE:
|
|
|
|
license++;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#ifdef NSIS_CONFIG_COMPONENTPAGE
|
|
|
|
case NSIS_PAGE_SELCOM:
|
|
|
|
selcom++;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case NSIS_PAGE_DIR:
|
|
|
|
dir++;
|
|
|
|
break;
|
|
|
|
case NSIS_PAGE_INSTFILES:
|
|
|
|
instlog++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
p++;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (license==1) {
|
2002-11-09 13:51:40 +00:00
|
|
|
ERROR_MSG("\nError: %s page and %s depend on each other, both must be in the script!\n", "license", "LicenseData");
|
2002-11-01 20:34:55 +00:00
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
if (selcom==1) {
|
2002-11-09 13:51:40 +00:00
|
|
|
ERROR_MSG("\nError: %s page and %s depend on each other, both must be in the script!\n", "components", "ComponentText");
|
2002-11-01 20:34:55 +00:00
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
if (dir==1) {
|
2002-11-09 13:51:40 +00:00
|
|
|
ERROR_MSG("\nError: %s page and %s depend on each other, both must be in the script!\n", "directory selection", "DirText");
|
2002-11-01 20:34:55 +00:00
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
if (!instlog) {
|
|
|
|
warning("Page instfiles not specefied, no sections will be executed!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
#ifdef NSIS_CONFIG_LICENSEPAGE
|
|
|
|
if (license) {
|
|
|
|
pg.id=NSIS_PAGE_LICENSE;
|
2002-11-11 19:19:02 +00:00
|
|
|
pg.caption=LANG_SUBCAPTION(0);
|
2002-11-01 20:34:55 +00:00
|
|
|
build_pages.add(&pg,sizeof(page));
|
|
|
|
build_header.common.num_pages++;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef NSIS_CONFIG_COMPONENTPAGE
|
|
|
|
if (selcom) {
|
|
|
|
pg.id=NSIS_PAGE_SELCOM;
|
2002-11-11 19:19:02 +00:00
|
|
|
pg.caption=LANG_SUBCAPTION(1);
|
2002-11-01 20:34:55 +00:00
|
|
|
build_pages.add(&pg,sizeof(page));
|
|
|
|
build_header.common.num_pages++;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
if (dir) {
|
|
|
|
pg.id=NSIS_PAGE_DIR;
|
2002-11-11 19:19:02 +00:00
|
|
|
pg.caption=LANG_SUBCAPTION(2);
|
2002-11-01 20:34:55 +00:00
|
|
|
build_pages.add(&pg,sizeof(page));
|
|
|
|
build_header.common.num_pages++;
|
|
|
|
}
|
|
|
|
instlog++;
|
|
|
|
pg.id=NSIS_PAGE_INSTFILES;
|
2002-11-11 19:19:02 +00:00
|
|
|
pg.caption=LANG_SUBCAPTION(3);
|
2002-11-01 20:34:55 +00:00
|
|
|
build_pages.add(&pg,sizeof(page));
|
|
|
|
build_header.common.num_pages++;
|
|
|
|
pg.id=NSIS_PAGE_COMPLETED;
|
2002-11-11 19:19:02 +00:00
|
|
|
pg.caption=LANG_SUBCAPTION(4);
|
2002-11-01 20:34:55 +00:00
|
|
|
build_pages.add(&pg,sizeof(page));
|
|
|
|
build_header.common.num_pages++;
|
|
|
|
}
|
|
|
|
|
|
|
|
page *p=(page *) build_pages.get();
|
|
|
|
for (int i=0; i<build_header.common.num_pages; i++, p++) {
|
2003-03-06 21:24:19 +00:00
|
|
|
// 2 - back enabled
|
|
|
|
// 4 - cancel enabled
|
|
|
|
// SW_SHOWNA (8) - back visible
|
|
|
|
if (i) p->button_states=SW_SHOWNA|2|4;
|
|
|
|
else p->button_states=4;
|
2002-11-01 20:34:55 +00:00
|
|
|
|
|
|
|
p->next=LANG_BTN_NEXT;
|
|
|
|
|
2002-11-25 16:50:05 +00:00
|
|
|
if (i<build_header.common.num_pages-1 && (p+1)->id==NSIS_PAGE_INSTFILES) {
|
2002-11-01 20:34:55 +00:00
|
|
|
p->next=LANG_BTN_INSTALL;
|
2002-11-25 16:50:05 +00:00
|
|
|
install_used = true;
|
|
|
|
}
|
2002-11-07 15:18:39 +00:00
|
|
|
#ifdef NSIS_CONFIG_LICENSEPAGE
|
2003-03-26 17:47:46 +00:00
|
|
|
if (p->id==NSIS_PAGE_LICENSE) {
|
|
|
|
if (build_header.common.flags&CH_FLAGS_LICENSE_FORCE_SELECTION)
|
|
|
|
p->button_states|=16;
|
|
|
|
else
|
|
|
|
p->next=LANG_BTN_LICENSE;
|
|
|
|
}
|
2002-11-07 15:18:39 +00:00
|
|
|
#endif
|
2002-11-01 20:34:55 +00:00
|
|
|
if (p->id==NSIS_PAGE_INSTFILES || p->id==NSIS_PAGE_COMPLETED)
|
2003-03-06 21:24:19 +00:00
|
|
|
p->button_states&=~6;
|
2002-11-09 20:19:13 +00:00
|
|
|
if (i && (p-1)->id==NSIS_PAGE_COMPLETED)
|
2003-03-06 21:24:19 +00:00
|
|
|
p->button_states&=~2;
|
2002-11-25 16:50:05 +00:00
|
|
|
|
|
|
|
if (p->next == LANG_BTN_NEXT) next_used = true;
|
2002-11-01 20:34:55 +00:00
|
|
|
}
|
|
|
|
(--p)->next=LANG_BTN_CLOSE;
|
2003-03-24 15:35:23 +00:00
|
|
|
if (!enable_last_page_cancel && instlog) {
|
2003-03-06 21:24:19 +00:00
|
|
|
p->button_states&=~4;
|
|
|
|
}
|
2002-11-01 20:34:55 +00:00
|
|
|
if (p->id==NSIS_PAGE_COMPLETED) (--p)->next=LANG_BTN_CLOSE;
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
#ifdef NSIS_CONFIG_SILENT_SUPPORT
|
2002-11-01 20:34:55 +00:00
|
|
|
else main--;
|
|
|
|
#endif
|
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
#ifdef NSIS_CONFIG_UNINSTALL_SUPPORT
|
|
|
|
#ifdef NSIS_CONFIG_SILENT_SUPPORT
|
2003-03-18 20:36:52 +00:00
|
|
|
if (!(build_header.common.flags&(CH_FLAGS_SILENT|CH_FLAGS_SILENT_LOG)) && uninstaller_writes_used)
|
2002-11-01 20:34:55 +00:00
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
2002-11-01 20:34:55 +00:00
|
|
|
if (!IsNotSet(uninstall.uninstalltext)) uninst++;
|
|
|
|
|
2003-03-24 15:35:23 +00:00
|
|
|
int uninstlog = 0;
|
|
|
|
|
2002-11-01 20:34:55 +00:00
|
|
|
if (!add_uninst_pages) {
|
|
|
|
int i=0;
|
|
|
|
page *p=(page *) ubuild_pages.get();
|
|
|
|
while (i!=build_header.common.num_pages) {
|
|
|
|
switch (p->id) {
|
2002-11-09 13:51:40 +00:00
|
|
|
#ifdef NSIS_CONFIG_UNINSTALL_SUPPORT
|
2002-11-01 20:34:55 +00:00
|
|
|
case NSIS_PAGE_UNINST:
|
|
|
|
uninst++;
|
|
|
|
break;
|
2002-11-09 13:51:40 +00:00
|
|
|
#endif
|
2002-11-01 20:34:55 +00:00
|
|
|
case NSIS_PAGE_INSTFILES:
|
|
|
|
instlog++;
|
2003-03-24 15:35:23 +00:00
|
|
|
uninstlog++;
|
2002-11-01 20:34:55 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
p++;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (uninst==1) {
|
2002-12-11 14:29:55 +00:00
|
|
|
ERROR_MSG("\nError: %s page and %s depend on each other, both must be in the script!\n", "uninstConfirm", "UninstallText");
|
2002-11-01 20:34:55 +00:00
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
if (!instlog) {
|
|
|
|
warning("UninstPage instfiles not specefied, no sections will be executed!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (uninst) {
|
|
|
|
pg.id=NSIS_PAGE_UNINST;
|
2002-11-11 19:19:02 +00:00
|
|
|
pg.caption=LANG_SUBCAPTION(0);
|
2002-11-01 20:34:55 +00:00
|
|
|
ubuild_pages.add(&pg,sizeof(page));
|
|
|
|
build_uninst.common.num_pages++;
|
|
|
|
}
|
|
|
|
instlog++;
|
|
|
|
pg.id=NSIS_PAGE_INSTFILES;
|
2002-11-11 19:19:02 +00:00
|
|
|
pg.caption=LANG_SUBCAPTION(1);
|
2002-11-01 20:34:55 +00:00
|
|
|
ubuild_pages.add(&pg,sizeof(page));
|
|
|
|
build_uninst.common.num_pages++;
|
|
|
|
pg.id=NSIS_PAGE_COMPLETED;
|
2002-11-11 19:19:02 +00:00
|
|
|
pg.caption=LANG_SUBCAPTION(2);
|
2002-11-01 20:34:55 +00:00
|
|
|
ubuild_pages.add(&pg,sizeof(page));
|
|
|
|
build_uninst.common.num_pages++;
|
|
|
|
}
|
|
|
|
|
|
|
|
page *p=(page *) ubuild_pages.get();
|
|
|
|
int noinstlogback=0;
|
|
|
|
for (int i=0; i<build_uninst.common.num_pages; i++, p++) {
|
2003-03-06 21:24:19 +00:00
|
|
|
// 2 - back enabled
|
|
|
|
// 4 - cancel enabled
|
|
|
|
// SW_SHOWNA (8) - back visible
|
|
|
|
if (i) p->button_states=SW_SHOWNA|2|4;
|
|
|
|
else p->button_states=4;
|
2002-11-01 20:34:55 +00:00
|
|
|
|
|
|
|
p->next=LANG_BTN_NEXT;
|
|
|
|
|
|
|
|
if (i<build_uninst.common.num_pages-1 && (p+1)->id==NSIS_PAGE_INSTFILES) {
|
|
|
|
if (p->id==NSIS_PAGE_UNINST)
|
|
|
|
noinstlogback=1;
|
|
|
|
p->next=LANG_BTN_UNINST;
|
|
|
|
}
|
|
|
|
if (p->id==NSIS_PAGE_INSTFILES || p->id==NSIS_PAGE_COMPLETED)
|
2003-03-06 21:24:19 +00:00
|
|
|
p->button_states=noinstlogback?0:SW_SHOWNA;
|
|
|
|
if (i && (p-1)->id==NSIS_PAGE_COMPLETED)
|
|
|
|
p->button_states&=~2;
|
2002-11-25 16:50:05 +00:00
|
|
|
|
|
|
|
if (p->next == LANG_BTN_NEXT) next_used = true;
|
2002-11-01 20:34:55 +00:00
|
|
|
}
|
|
|
|
(--p)->next=LANG_BTN_CLOSE;
|
2003-03-24 15:35:23 +00:00
|
|
|
if (!uenable_last_page_cancel && uninstlog) {
|
2003-03-06 21:24:19 +00:00
|
|
|
p->button_states&=~4;
|
|
|
|
}
|
2002-11-01 20:34:55 +00:00
|
|
|
if (p->id==NSIS_PAGE_COMPLETED) (--p)->next=LANG_BTN_CLOSE;
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
#ifdef NSIS_CONFIG_SILENT_SUPPORT
|
2002-11-01 20:34:55 +00:00
|
|
|
else
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
main--;
|
2002-11-09 13:51:40 +00:00
|
|
|
|
|
|
|
SCRIPT_MSG("Done!\n");
|
2002-11-01 20:34:55 +00:00
|
|
|
|
|
|
|
try {
|
|
|
|
SCRIPT_MSG("Removing unused resources... ");
|
2002-11-09 13:51:40 +00:00
|
|
|
init_res_editor();
|
|
|
|
#ifdef NSIS_CONFIG_LICENSEPAGE
|
2002-11-01 20:34:55 +00:00
|
|
|
if (!license) {
|
2002-11-09 13:51:40 +00:00
|
|
|
res_editor->UpdateResource(RT_DIALOG, IDD_LICENSE, MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), 0, 0);
|
2002-11-01 20:34:55 +00:00
|
|
|
}
|
2002-11-09 13:51:40 +00:00
|
|
|
#endif // NSIS_CONFIG_LICENSEPAGE
|
|
|
|
#ifdef NSIS_CONFIG_COMPONENTPAGE
|
2002-11-01 20:34:55 +00:00
|
|
|
if (!selcom) {
|
2002-11-09 13:51:40 +00:00
|
|
|
res_editor->UpdateResource(RT_DIALOG, IDD_SELCOM, MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), 0, 0);
|
|
|
|
res_editor->UpdateResource(RT_BITMAP, IDB_BITMAP1, MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), 0, 0);
|
2002-11-01 20:34:55 +00:00
|
|
|
}
|
2002-11-09 13:51:40 +00:00
|
|
|
#endif // NSIS_CONFIG_COMPONENTPAGE
|
2002-11-01 20:34:55 +00:00
|
|
|
if (!dir) {
|
2002-11-09 13:51:40 +00:00
|
|
|
res_editor->UpdateResource(RT_DIALOG, IDD_DIR, MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), 0, 0);
|
2002-11-01 20:34:55 +00:00
|
|
|
}
|
2002-11-09 13:51:40 +00:00
|
|
|
#ifdef NSIS_CONFIG_UNINSTALL_SUPPORT
|
2002-11-01 20:34:55 +00:00
|
|
|
if (!uninst) {
|
2002-11-09 13:51:40 +00:00
|
|
|
res_editor->UpdateResource(RT_DIALOG, IDD_UNINST, MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), 0, 0);
|
2002-11-01 20:34:55 +00:00
|
|
|
}
|
2002-11-09 13:51:40 +00:00
|
|
|
#endif // NSIS_CONFIG_UNINSTALL_SUPPORT
|
2002-11-01 20:34:55 +00:00
|
|
|
if (!instlog) {
|
2002-11-09 13:51:40 +00:00
|
|
|
res_editor->UpdateResource(RT_DIALOG, IDD_INSTFILES, MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), 0, 0);
|
2002-11-01 20:34:55 +00:00
|
|
|
}
|
|
|
|
if (!main) {
|
2002-11-09 13:51:40 +00:00
|
|
|
res_editor->UpdateResource(RT_DIALOG, IDD_INST, MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), 0, 0);
|
2002-11-01 20:34:55 +00:00
|
|
|
if (!build_compress_whole && !build_crcchk)
|
2002-11-09 13:51:40 +00:00
|
|
|
res_editor->UpdateResource(RT_DIALOG, IDD_VERIFY, MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), 0, 0);
|
2002-11-01 20:34:55 +00:00
|
|
|
}
|
2002-08-02 10:01:35 +00:00
|
|
|
|
2002-11-01 20:34:55 +00:00
|
|
|
SCRIPT_MSG("Done!\n");
|
|
|
|
}
|
|
|
|
catch (exception& err) {
|
|
|
|
ERROR_MSG("\nError: %s\n", err.what());
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
#endif // NSIS_CONFIG_VISIBLE_SUPPORT
|
2002-08-02 16:51:42 +00:00
|
|
|
|
2003-04-21 13:32:34 +00:00
|
|
|
// Generate language tables
|
|
|
|
if (WriteStringTables() == PS_ERROR) return PS_ERROR;
|
|
|
|
|
2002-11-09 13:51:40 +00:00
|
|
|
// Save all changes to the exe header
|
|
|
|
try {
|
|
|
|
close_res_editor();
|
|
|
|
}
|
|
|
|
catch (exception& err) {
|
|
|
|
ERROR_MSG("\nError: %s\n", err.what());
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
|
2002-08-02 16:51:42 +00:00
|
|
|
// Pack exe header if asked for
|
2002-08-02 10:01:35 +00:00
|
|
|
if (build_packname[0] && build_packcmd[0])
|
|
|
|
{
|
|
|
|
FILE *tmpfile=fopen(build_packname,"wb");
|
|
|
|
if (!tmpfile)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: writing temporary file \"%s\" for pack\n",build_packname);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
fwrite(header_data_new,1,exeheader_size_new,tmpfile);
|
|
|
|
fclose(tmpfile);
|
|
|
|
if (system(build_packcmd) == -1)
|
|
|
|
{
|
|
|
|
remove(build_packname);
|
|
|
|
ERROR_MSG("Error: calling packer on \"%s\"\n",build_packname);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
tmpfile=fopen(build_packname,"rb");
|
|
|
|
if (!tmpfile)
|
|
|
|
{
|
|
|
|
remove(build_packname);
|
|
|
|
ERROR_MSG("Error: reading temporary file \"%s\" after pack\n",build_packname);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
fseek(tmpfile,0,SEEK_END);
|
|
|
|
exeheader_size_new=ftell(tmpfile);
|
|
|
|
exeheader_size_new+=511;
|
|
|
|
exeheader_size_new&=~511; // align to 512.
|
|
|
|
fseek(tmpfile,0,SEEK_SET);
|
|
|
|
unsigned char *header_data_older=header_data_new;
|
|
|
|
header_data_new=(unsigned char *)malloc(exeheader_size_new);
|
|
|
|
if (!header_data_new)
|
|
|
|
{
|
|
|
|
free(header_data_older);
|
|
|
|
fclose(tmpfile);
|
|
|
|
ERROR_MSG("Error: malloc(%d) failed (exepack)\n",exeheader_size_new);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
memset(header_data_new,0,exeheader_size_new);
|
|
|
|
fread(header_data_new,1,exeheader_size_new,tmpfile);
|
|
|
|
fclose(tmpfile);
|
|
|
|
remove(build_packname);
|
|
|
|
|
|
|
|
// write out exe header, pack, read back in, align to 512, and
|
|
|
|
// update the header info
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef NSIS_CONFIG_UNINSTALL_SUPPORT
|
|
|
|
// Get offsets of icons to replace for uninstall
|
|
|
|
// Also makes sure that the icons are there and in the right size.
|
|
|
|
if (uninstaller_writes_used) {
|
|
|
|
SCRIPT_MSG("Finding icons offsets for uninstaller... ");
|
|
|
|
icon_offset = generate_unicons_offsets(header_data_new, m_unicon_data);
|
|
|
|
if (icon_offset == 0)
|
|
|
|
return PS_ERROR;
|
|
|
|
SCRIPT_MSG("Done!\n");
|
|
|
|
}
|
|
|
|
#endif //NSIS_CONFIG_UNINSTALL_SUPPORT
|
|
|
|
|
|
|
|
build_optimize_datablock=0;
|
|
|
|
|
2002-08-31 23:44:58 +00:00
|
|
|
if (uninstall_generate() != PS_OK)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
int crc=0;
|
|
|
|
|
|
|
|
{
|
|
|
|
char buffer[1024],*p;
|
|
|
|
GetFullPathName(build_output_filename,1024,buffer,&p);
|
2002-12-24 20:35:26 +00:00
|
|
|
notify(MAKENSIS_NOTIFY_OUTPUT, buffer);
|
2002-08-02 10:01:35 +00:00
|
|
|
INFO_MSG("\nOutput: \"%s\"\n",buffer);
|
|
|
|
}
|
|
|
|
FILE *fp = fopen(build_output_filename,"w+b");
|
2002-08-31 23:44:58 +00:00
|
|
|
if (!fp)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
ERROR_MSG("Can't open output file\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((int)fwrite(header_data_new,1,exeheader_size_new,fp) != exeheader_size_new)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: can't write %d bytes to output\n",exeheader_size_new);
|
|
|
|
fclose(fp);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
#ifdef NSIS_CONFIG_CRC_SUPPORT
|
|
|
|
#ifdef NSIS_CONFIG_CRC_ANAL
|
|
|
|
crc=CRC32(crc,header_data_new,exeheader_size_new);
|
|
|
|
#else
|
|
|
|
crc=CRC32(crc,header_data_new+512,exeheader_size_new-512);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
firstheader fh={0,};
|
|
|
|
fh.nsinst[0]=FH_INT1;
|
|
|
|
fh.nsinst[1]=FH_INT2;
|
|
|
|
fh.nsinst[2]=FH_INT3;
|
|
|
|
|
|
|
|
fh.flags=(build_crcchk?FH_FLAGS_CRC:0);
|
|
|
|
fh.flags|=(build_crcchk==2?FH_FLAGS_FORCE_CRC:0);
|
|
|
|
#ifdef NSIS_CONFIG_SILENT_SUPPORT
|
2003-03-18 20:36:52 +00:00
|
|
|
if (build_header.common.flags&(CH_FLAGS_SILENT|CH_FLAGS_SILENT_LOG)) fh.flags |= FH_FLAGS_SILENT;
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
|
|
|
fh.siginfo=FH_SIG;
|
|
|
|
|
|
|
|
int installinfo_compressed;
|
|
|
|
int fd_start;
|
2002-09-18 19:08:53 +00:00
|
|
|
|
|
|
|
#ifdef NSIS_CONFIG_COMPRESSION_SUPPORT
|
2002-08-02 10:01:35 +00:00
|
|
|
if (build_compress_whole)
|
|
|
|
{
|
|
|
|
if ((compressor->Init(9)) != C_OK)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: deflateInit() returned < 0\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
#endif
|
|
|
|
|
2002-09-24 10:38:55 +00:00
|
|
|
build_header.common.num_string_bytes=build_strlist.getlen();
|
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
GrowBuf ihd;
|
|
|
|
{
|
|
|
|
GrowBuf hdrcomp;
|
2002-08-31 23:44:58 +00:00
|
|
|
hdrcomp.add(&build_header,sizeof(build_header));
|
|
|
|
hdrcomp.add(build_sections.get(),build_sections.getlen());
|
2002-11-01 20:34:55 +00:00
|
|
|
#ifdef NSIS_CONFIG_VISIBLE_SUPPORT
|
|
|
|
hdrcomp.add(build_pages.get(),build_pages.getlen());
|
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
hdrcomp.add(build_entries.get(),build_entries.getlen());
|
|
|
|
hdrcomp.add(build_strlist.get(),build_strlist.getlen());
|
2002-09-20 23:22:11 +00:00
|
|
|
hdrcomp.add(build_langtables.get(),build_langtables.getlen());
|
2002-08-02 10:01:35 +00:00
|
|
|
|
|
|
|
if (add_data((char*)hdrcomp.get(),hdrcomp.getlen(),&ihd) < 0) return PS_ERROR;
|
2002-08-31 23:44:58 +00:00
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
installinfo_compressed=ihd.getlen();
|
|
|
|
fh.length_of_header=hdrcomp.getlen();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!build_compress_whole)
|
|
|
|
fh.length_of_all_following_data=ihd.getlen()+build_datablock.getlen()+(int)sizeof(firstheader)+(build_crcchk?sizeof(int):0);
|
|
|
|
else
|
|
|
|
fd_start=ftell(fp);
|
|
|
|
|
|
|
|
if (fwrite(&fh,1,sizeof(fh),fp) != sizeof(fh))
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: can't write %d bytes to output\n",sizeof(fh));
|
|
|
|
fclose(fp);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
|
2002-09-18 19:08:53 +00:00
|
|
|
#ifdef NSIS_CONFIG_COMPRESSION_SUPPORT
|
2002-08-02 10:01:35 +00:00
|
|
|
if (build_compress_whole) {
|
|
|
|
if (deflateToFile(fp,(char*)ihd.get(),ihd.getlen()))
|
|
|
|
{
|
|
|
|
fclose(fp);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
else
|
|
|
|
#endif
|
|
|
|
{
|
2002-08-02 10:01:35 +00:00
|
|
|
if (fwrite(ihd.get(),1,ihd.getlen(),fp) != (unsigned int)ihd.getlen())
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: can't write %d bytes to output\n",ihd.getlen());
|
|
|
|
fclose(fp);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
#ifdef NSIS_CONFIG_CRC_SUPPORT
|
|
|
|
crc=CRC32(crc,(unsigned char*)&fh,sizeof(fh));
|
|
|
|
crc=CRC32(crc,(unsigned char*)ihd.get(),ihd.getlen());
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
int ns=build_sections.getlen()/sizeof(section);
|
|
|
|
section *s=(section*)build_sections.get();
|
|
|
|
int x;
|
2002-08-22 21:06:07 +00:00
|
|
|
int req=0;
|
2002-08-02 10:01:35 +00:00
|
|
|
for (x = 1; x < ns; x ++)
|
|
|
|
{
|
2002-10-02 15:01:06 +00:00
|
|
|
if (!s[x].name_ptr || s[x].flags & SF_RO) req++;
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
INFO_MSG("Install: %d section%s",ns,ns==1?"":"s");
|
2002-08-22 21:06:07 +00:00
|
|
|
if (req)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
2002-08-22 21:06:07 +00:00
|
|
|
INFO_MSG(" (%d required)",req);
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
INFO_MSG(".\n");
|
|
|
|
}
|
|
|
|
int ne=build_entries.getlen()/sizeof(entry);
|
|
|
|
INFO_MSG("Install: %d instruction%s (%d bytes), ",ne,ne==1?"":"s",ne*sizeof(entry));
|
2002-11-09 13:08:45 +00:00
|
|
|
int ns=build_strlist.getnum();
|
|
|
|
INFO_MSG("%d string%s (%d bytes), ",ns,ns==1?"":"s",build_strlist.getlen());
|
|
|
|
int nlt=string_tables.size();
|
|
|
|
INFO_MSG("%d language table%s (%d bytes), ",nlt,nlt==1?"":"s",build_langtables.getlen());
|
|
|
|
int np=build_pages.getlen()/sizeof(page);
|
|
|
|
INFO_MSG("%d page%s (%d bytes).\n",np,np==1?"":"s",np*sizeof(page));
|
2002-08-31 23:44:58 +00:00
|
|
|
if (ubuild_entries.getlen())
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
2002-11-09 13:08:45 +00:00
|
|
|
ne=ubuild_entries.getlen()/sizeof(entry);
|
|
|
|
INFO_MSG("Uninstall: %d instruction%s (%d bytes), ",ne,ne==1?"":"s",ne*sizeof(entry));
|
|
|
|
ns=ubuild_strlist.getnum();
|
|
|
|
INFO_MSG("%d string%s (%d bytes), ",ns,ns==1?"":"s",ubuild_strlist.getlen());
|
|
|
|
nlt=string_tables.size();
|
|
|
|
INFO_MSG("%d language table%s (%d bytes), ",nlt,nlt==1?"":"s",ubuild_langtables.getlen());
|
|
|
|
np=ubuild_pages.getlen()/sizeof(page);
|
|
|
|
INFO_MSG("%d page%s (%d bytes).\n",np,np==1?"":"s",np*sizeof(page));
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-08-31 23:44:58 +00:00
|
|
|
if (db_opt_save)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
int total_out_size_estimate=
|
|
|
|
exeheader_size_new+sizeof(fh)+build_datablock.getlen()+(build_crcchk?4:0);
|
|
|
|
int pc=MulDiv(db_opt_save,1000,db_opt_save+total_out_size_estimate);
|
|
|
|
INFO_MSG("Datablock optimizer saved %d bytes (~%d.%d%%).\n",db_opt_save,
|
|
|
|
pc/10,pc%10);
|
|
|
|
}
|
2003-02-10 18:43:40 +00:00
|
|
|
|
|
|
|
#ifdef NSIS_CONFIG_COMPRESSION_SUPPORT
|
2003-02-07 23:04:25 +00:00
|
|
|
INFO_MSG("\nUsing %s%s compression.\n\n", compressor->GetName(), build_compress_whole?" (compress whole)":"");
|
2003-02-10 18:43:40 +00:00
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
|
|
|
|
int total_usize=exeheader_size;
|
|
|
|
|
|
|
|
INFO_MSG("EXE header size: %10d / %d bytes\n",exeheader_size_new,exeheader_size);
|
|
|
|
|
|
|
|
if (build_compress_whole) {
|
2002-10-02 04:32:10 +00:00
|
|
|
INFO_MSG("Install code: (%d bytes)\n",
|
2002-08-31 23:44:58 +00:00
|
|
|
sizeof(fh)+fh.length_of_header+sizeof(int));
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
else {
|
2002-10-02 04:32:10 +00:00
|
|
|
INFO_MSG("Install code: %10d / %d bytes\n",
|
2002-08-02 10:01:35 +00:00
|
|
|
sizeof(fh)+installinfo_compressed,
|
2002-08-31 23:44:58 +00:00
|
|
|
sizeof(fh)+fh.length_of_header+sizeof(int));
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
total_usize+=sizeof(fh)+fh.length_of_header+sizeof(int);
|
|
|
|
|
|
|
|
{
|
|
|
|
int dbsize, dbsizeu;
|
|
|
|
dbsize = build_datablock.getlen();
|
|
|
|
if (uninstall_size>0) dbsize-=uninstall_size;
|
|
|
|
|
|
|
|
if (build_compress_whole) {
|
|
|
|
dbsizeu=dbsize;
|
|
|
|
INFO_MSG("Install data: (%d bytes)\n",dbsizeu);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dbsizeu = db_full_size - uninstall_size_full;
|
|
|
|
INFO_MSG("Install data: %10d / %d bytes\n",dbsize,dbsizeu);
|
|
|
|
}
|
|
|
|
total_usize+=dbsizeu;
|
|
|
|
}
|
|
|
|
|
2002-08-31 23:44:58 +00:00
|
|
|
if (uninstall_size>=0)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
if (build_compress_whole)
|
2002-10-02 04:32:10 +00:00
|
|
|
INFO_MSG("Uninstall code+data: (%d bytes)\n",uninstall_size_full);
|
2002-08-02 10:01:35 +00:00
|
|
|
else
|
2002-10-02 04:32:10 +00:00
|
|
|
INFO_MSG("Uninstall code+data: %6d / %d bytes\n",uninstall_size,uninstall_size_full);
|
2002-08-02 10:01:35 +00:00
|
|
|
total_usize+=uninstall_size_full;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (build_datablock.getlen())
|
|
|
|
{
|
|
|
|
char *dbptr=(char *)build_datablock.get();
|
|
|
|
int dbl=build_datablock.getlen();
|
|
|
|
if (build_compress_whole) {
|
|
|
|
INFO_MSG("Compressed data: ");
|
|
|
|
}
|
|
|
|
while (dbl > 0)
|
|
|
|
{
|
|
|
|
int l=dbl;
|
|
|
|
if (l > 32768) l=32768;
|
2002-09-18 19:08:53 +00:00
|
|
|
#ifdef NSIS_CONFIG_COMPRESSION_SUPPORT
|
2002-08-02 10:01:35 +00:00
|
|
|
if (build_compress_whole) {
|
|
|
|
if (deflateToFile(fp,dbptr,l))
|
|
|
|
{
|
|
|
|
fclose(fp);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
else
|
|
|
|
#endif
|
|
|
|
{
|
2002-08-02 10:01:35 +00:00
|
|
|
#ifdef NSIS_CONFIG_CRC_SUPPORT
|
|
|
|
crc=CRC32(crc,(unsigned char *)dbptr,l);
|
|
|
|
#endif
|
|
|
|
if ((int)fwrite(dbptr,1,l,fp) != l)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: can't write %d bytes to output\n",l);
|
|
|
|
fclose(fp);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
dbptr+=l;
|
|
|
|
dbl-=l;
|
|
|
|
}
|
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
#ifdef NSIS_CONFIG_COMPRESSION_SUPPORT
|
|
|
|
if (build_compress_whole)
|
|
|
|
{
|
2002-08-02 10:01:35 +00:00
|
|
|
if (deflateToFile(fp,NULL,0))
|
|
|
|
{
|
|
|
|
fclose(fp);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
#ifdef NSIS_CONFIG_COMPRESSION_SUPPORT
|
2002-08-02 10:01:35 +00:00
|
|
|
compressor->End();
|
2002-09-18 19:08:53 +00:00
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
|
|
|
|
fh.length_of_all_following_data=(ftell(fp)-fd_start)+(build_crcchk?sizeof(int):0);
|
|
|
|
INFO_MSG("%10d / %d bytes\n",(ftell(fp)-fd_start),build_datablock.getlen());
|
|
|
|
|
|
|
|
fseek(fp,fd_start,SEEK_SET);
|
|
|
|
fwrite(&fh,sizeof(fh),1,fp);
|
|
|
|
|
|
|
|
#ifdef NSIS_CONFIG_CRC_SUPPORT
|
|
|
|
if (build_crcchk)
|
|
|
|
{
|
|
|
|
// check rest of CRC
|
|
|
|
fseek(fp,fd_start,SEEK_SET);
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
char buf[32768];
|
|
|
|
int l=fread(buf,1,sizeof(buf),fp);
|
|
|
|
if (!l) break;
|
|
|
|
crc=CRC32(crc,(unsigned char *)buf,l);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
fseek(fp,0,SEEK_END); // reset eof flag
|
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
|
2002-08-31 23:44:58 +00:00
|
|
|
if (build_crcchk)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
total_usize+=sizeof(int);
|
|
|
|
if (fwrite(&crc,1,sizeof(int),fp) != sizeof(int))
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: can't write %d bytes to output\n",sizeof(int));
|
|
|
|
fclose(fp);
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
INFO_MSG("CRC (0x%08X): 4 / 4 bytes\n",crc);
|
|
|
|
}
|
|
|
|
INFO_MSG("\n");
|
|
|
|
{
|
|
|
|
int pc=MulDiv(ftell(fp),1000,total_usize);
|
|
|
|
INFO_MSG("Total size: %10d / %d bytes (%d.%d%%)\n",
|
|
|
|
ftell(fp),total_usize,pc/10,pc%10);
|
|
|
|
}
|
|
|
|
fclose(fp);
|
|
|
|
print_warnings();
|
|
|
|
return PS_OK;
|
|
|
|
}
|
|
|
|
|
2002-09-18 19:08:53 +00:00
|
|
|
#ifdef NSIS_CONFIG_COMPRESSION_SUPPORT
|
2002-08-02 10:01:35 +00:00
|
|
|
int CEXEBuild::deflateToFile(FILE *fp, char *buf, int len) // len==0 to flush
|
|
|
|
{
|
|
|
|
build_compressor_set=true;
|
|
|
|
|
|
|
|
char obuf[32768];
|
|
|
|
int flush=0;
|
|
|
|
compressor->SetNextIn(buf,len);
|
2002-08-31 23:44:58 +00:00
|
|
|
if (!buf||!len)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
char a;
|
|
|
|
compressor->SetNextIn(&a,0);
|
|
|
|
flush=C_FINISH;
|
|
|
|
}
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
compressor->SetNextOut(obuf,sizeof(obuf));
|
|
|
|
int ret=compressor->Compress(flush);
|
|
|
|
if (ret<0 && (ret!=-1 || !flush))
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: deflateToFile: deflate()=%d\n",ret);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
int l=compressor->GetNextOut()-obuf;
|
|
|
|
if (l)
|
|
|
|
{
|
|
|
|
if (fwrite(obuf,1,l,fp) != (unsigned)l)
|
|
|
|
{
|
|
|
|
ERROR_MSG("Error: deflateToFile fwrite(%d) failed\n",l);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!compressor->GetAvailIn() && (!flush || !l)) break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2002-09-18 19:08:53 +00:00
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
|
|
|
|
int CEXEBuild::uninstall_generate()
|
|
|
|
{
|
|
|
|
#ifdef NSIS_CONFIG_UNINSTALL_SUPPORT
|
2002-08-31 23:44:58 +00:00
|
|
|
if (ubuild_entries.getlen() && uninstaller_writes_used)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
firstheader fh={0,};
|
|
|
|
|
2002-09-20 23:22:11 +00:00
|
|
|
build_uninst.common.num_string_bytes=ubuild_strlist.getlen();
|
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
GrowBuf uhd;
|
|
|
|
// add one more bit (the code+strtabs) to the uninstall datablock
|
|
|
|
{
|
|
|
|
GrowBuf udata;
|
|
|
|
|
|
|
|
udata.add(&build_uninst,sizeof(build_uninst));
|
2002-11-01 20:34:55 +00:00
|
|
|
#ifdef NSIS_CONFIG_VISIBLE_SUPPORT
|
|
|
|
udata.add(ubuild_pages.get(),ubuild_pages.getlen());
|
|
|
|
#endif
|
2002-08-02 10:01:35 +00:00
|
|
|
udata.add(ubuild_entries.get(),ubuild_entries.getlen());
|
2002-08-31 23:44:58 +00:00
|
|
|
udata.add(ubuild_strlist.get(),ubuild_strlist.getlen());
|
2002-09-20 23:22:11 +00:00
|
|
|
udata.add(ubuild_langtables.get(),ubuild_langtables.getlen());
|
2002-08-31 23:44:58 +00:00
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
set_uninstall_mode(1);
|
|
|
|
fh.length_of_header=udata.getlen();
|
2002-08-31 23:44:58 +00:00
|
|
|
int err=add_data((char*)udata.get(),udata.getlen(),&uhd);
|
2002-08-02 10:01:35 +00:00
|
|
|
set_uninstall_mode(0);
|
|
|
|
if (err < 0) return PS_ERROR;
|
|
|
|
}
|
2002-08-31 23:44:58 +00:00
|
|
|
|
2002-08-02 10:01:35 +00:00
|
|
|
int crc=0;
|
|
|
|
|
|
|
|
build_header.uninstdata_offset=build_datablock.getlen();
|
|
|
|
build_header.uninsticon_size=unicondata_size;
|
|
|
|
|
|
|
|
if (add_data((char *)m_unicon_data,unicondata_size) < 0)
|
|
|
|
return PS_ERROR;
|
|
|
|
#ifdef NSIS_CONFIG_CRC_SUPPORT
|
|
|
|
#ifdef NSIS_CONFIG_CRC_ANAL
|
|
|
|
crc=CRC32(crc,header_data_new,icon_offset);
|
|
|
|
#else
|
|
|
|
crc=CRC32(crc,header_data_new+512,icon_offset-512);
|
|
|
|
#endif
|
|
|
|
// Changed by Amir Szekely 11th July 2002
|
|
|
|
// This bunch of lines do CRC for the uninstaller icon data
|
2002-09-23 14:07:39 +00:00
|
|
|
unsigned char* seeker = m_unicon_data;
|
2002-08-02 10:01:35 +00:00
|
|
|
DWORD dwEndOfIcons = 0;
|
2002-09-23 14:07:39 +00:00
|
|
|
while (*seeker) {
|
2002-08-02 10:01:35 +00:00
|
|
|
DWORD dwSize = *(DWORD*)seeker;
|
|
|
|
seeker += sizeof(DWORD);
|
|
|
|
DWORD dwOffset = *(DWORD*)seeker;
|
|
|
|
seeker += sizeof(DWORD);
|
|
|
|
// Do CRC for icon data
|
|
|
|
crc=CRC32(crc,seeker,dwSize);
|
|
|
|
seeker += dwSize;
|
2002-09-23 14:07:39 +00:00
|
|
|
if (*seeker) {
|
2002-08-02 10:01:35 +00:00
|
|
|
// Do CRC for data between icons
|
|
|
|
crc=CRC32(crc,header_data_new+dwOffset+dwSize,(*(DWORD*)(seeker+sizeof(DWORD)))-dwOffset-dwSize);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dwEndOfIcons = dwOffset+dwSize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Done with icons, carry on
|
|
|
|
crc=CRC32(crc,header_data_new+dwEndOfIcons,exeheader_size_new-dwEndOfIcons);
|
|
|
|
#endif
|
|
|
|
fh.nsinst[0]=FH_INT1;
|
|
|
|
fh.nsinst[1]=FH_INT2;
|
|
|
|
fh.nsinst[2]=FH_INT3;
|
|
|
|
fh.flags = FH_FLAGS_UNINSTALL | (build_crcchk?FH_FLAGS_CRC:0);
|
|
|
|
fh.flags |= (build_crcchk==2?FH_FLAGS_FORCE_CRC:0);
|
|
|
|
#ifdef NSIS_CONFIG_SILENT_SUPPORT
|
2003-03-18 20:36:52 +00:00
|
|
|
if (build_uninst.common.flags&(CH_FLAGS_SILENT|CH_FLAGS_SILENT_LOG)) fh.flags |= FH_FLAGS_SILENT;
|
2002-08-02 10:01:35 +00:00
|
|
|
#endif
|
|
|
|
fh.siginfo=FH_SIG;
|
|
|
|
fh.length_of_all_following_data=
|
|
|
|
uhd.getlen()+ubuild_datablock.getlen()+(int)sizeof(firstheader)+(build_crcchk?sizeof(int):0);
|
|
|
|
|
|
|
|
GrowBuf udata;
|
|
|
|
|
2002-09-22 18:48:58 +00:00
|
|
|
#ifdef NSIS_CONFIG_COMPRESSION_SUPPORT
|
2002-08-02 10:01:35 +00:00
|
|
|
if (build_compress_whole) {
|
|
|
|
// compress uninstaller too
|
|
|
|
udata.add(&fh,sizeof(fh));
|
|
|
|
{
|
|
|
|
char obuf[32768];
|
2002-08-31 23:44:58 +00:00
|
|
|
if ((compressor->Init(9)) != C_OK)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
ERROR_MSG("Error: deflateInit() returned < 0\n");
|
|
|
|
return PS_ERROR;
|
|
|
|
}
|
|
|
|
int x;
|
|
|
|
for (x = 0; x < 2; x ++)
|
|
|
|
{
|
|
|
|
if (!x)
|
|
|
|
compressor->SetNextIn((char*)uhd.get(),uhd.getlen());
|
|
|
|
else
|
|
|
|
compressor->SetNextIn((char*)ubuild_datablock.get(),ubuild_datablock.getlen());
|
|
|
|
while (compressor->GetAvailIn())
|
|
|
|
{
|
|
|
|
compressor->SetNextOut(obuf,sizeof(obuf));
|
|
|
|
compressor->Compress(0);
|
|
|
|
if (compressor->GetNextOut()-obuf > 0)
|
|
|
|
udata.add(obuf,compressor->GetNextOut()-obuf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
compressor->SetNextOut(obuf,sizeof(obuf));
|
|
|
|
compressor->Compress(C_FINISH);
|
|
|
|
if (compressor->GetNextOut()-obuf > 0)
|
|
|
|
udata.add(obuf,compressor->GetNextOut()-obuf);
|
|
|
|
else break;
|
2002-08-31 23:44:58 +00:00
|
|
|
}
|
2002-08-02 10:01:35 +00:00
|
|
|
compressor->End();
|
|
|
|
}
|
|
|
|
|
|
|
|
firstheader *_fh=(firstheader *)udata.get();
|
|
|
|
_fh->length_of_all_following_data=udata.getlen()+(build_crcchk?sizeof(int):0);
|
|
|
|
}
|
2003-02-07 23:04:25 +00:00
|
|
|
else
|
2002-09-22 18:48:58 +00:00
|
|
|
#endif//NSIS_CONFIG_COMPRESSION_SUPPORT
|
|
|
|
{
|
2002-08-02 10:01:35 +00:00
|
|
|
udata.add(&fh,sizeof(fh));
|
|
|
|
udata.add(uhd.get(),uhd.getlen());
|
|
|
|
udata.add(ubuild_datablock.get(),ubuild_datablock.getlen());
|
|
|
|
}
|
|
|
|
#ifdef NSIS_CONFIG_CRC_SUPPORT
|
|
|
|
if (build_crcchk)
|
|
|
|
{
|
|
|
|
int s=CRC32(crc,(unsigned char *)udata.get(),udata.getlen());
|
|
|
|
udata.add(&s,sizeof(int));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2002-08-31 23:44:58 +00:00
|
|
|
if (add_data((char*)udata.get(),udata.getlen()) < 0)
|
2002-08-02 10:01:35 +00:00
|
|
|
return PS_ERROR;
|
|
|
|
|
|
|
|
uninstall_size_full=fh.length_of_all_following_data + sizeof(int) + unicondata_size - 32 + sizeof(int);
|
|
|
|
|
|
|
|
// compressed size
|
|
|
|
uninstall_size=build_datablock.getlen()-build_header.uninstdata_offset;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return PS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define SWAP(x,y,i) { i _ii; _ii=x; x=y; y=_ii; }
|
|
|
|
|
|
|
|
void CEXEBuild::set_uninstall_mode(int un)
|
|
|
|
{
|
|
|
|
if (un != uninstall_mode)
|
|
|
|
{
|
|
|
|
uninstall_mode=un;
|
2002-09-29 20:25:15 +00:00
|
|
|
if (un)
|
|
|
|
{
|
|
|
|
cur_datablock=&ubuild_datablock;
|
|
|
|
cur_entries=&ubuild_entries;
|
|
|
|
cur_functions=&ubuild_functions;
|
|
|
|
cur_labels=&ubuild_labels;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cur_datablock=&build_datablock;
|
|
|
|
cur_entries=&build_entries;
|
|
|
|
cur_functions=&build_functions;
|
|
|
|
cur_labels=&build_labels;
|
|
|
|
}
|
2002-08-02 10:01:35 +00:00
|
|
|
|
|
|
|
SWAP(db_opt_save_u,db_opt_save,int);
|
|
|
|
SWAP(db_comp_save_u,db_comp_save,int);
|
|
|
|
SWAP(db_full_size_u,db_full_size,int);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern FILE *g_output;
|
|
|
|
|
|
|
|
void CEXEBuild::warning(const char *s, ...)
|
|
|
|
{
|
|
|
|
char buf[4096];
|
|
|
|
va_list val;
|
|
|
|
va_start(val,s);
|
|
|
|
vsprintf(buf,s,val);
|
|
|
|
va_end(val);
|
|
|
|
m_warnings.add(buf,-1);
|
2002-12-24 20:35:26 +00:00
|
|
|
notify(MAKENSIS_NOTIFY_WARNING,buf);
|
2002-08-31 23:44:58 +00:00
|
|
|
if (display_warnings)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
fprintf(g_output,"warning: %s\n",buf);
|
|
|
|
fflush(g_output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CEXEBuild::ERROR_MSG(const char *s, ...)
|
|
|
|
{
|
2002-12-24 20:35:26 +00:00
|
|
|
if (display_errors || notify_hwnd)
|
2002-08-02 10:01:35 +00:00
|
|
|
{
|
|
|
|
char buf[4096];
|
|
|
|
va_list val;
|
|
|
|
va_start(val,s);
|
|
|
|
vsprintf(buf,s,val);
|
|
|
|
va_end(val);
|
2002-12-24 20:35:26 +00:00
|
|
|
notify(MAKENSIS_NOTIFY_ERROR,buf);
|
|
|
|
if (display_errors)
|
|
|
|
{
|
|
|
|
fprintf(g_output,"%s",buf);
|
|
|
|
fflush(g_output);
|
|
|
|
}
|
2002-08-02 10:01:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CEXEBuild::SCRIPT_MSG(const char *s, ...)
|
|
|
|
{
|
|
|
|
if (display_script)
|
|
|
|
{
|
|
|
|
char buf[4096];
|
|
|
|
va_list val;
|
|
|
|
va_start(val,s);
|
|
|
|
vsprintf(buf,s,val);
|
|
|
|
va_end(val);
|
|
|
|
fprintf(g_output,"%s",buf);
|
|
|
|
fflush(g_output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CEXEBuild::INFO_MSG(const char *s, ...)
|
|
|
|
{
|
|
|
|
if (display_info)
|
|
|
|
{
|
|
|
|
char buf[4096];
|
|
|
|
va_list val;
|
|
|
|
va_start(val,s);
|
|
|
|
vsprintf(buf,s,val);
|
|
|
|
va_end(val);
|
|
|
|
fprintf(g_output,"%s",buf);
|
|
|
|
fflush(g_output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CEXEBuild::print_warnings()
|
|
|
|
{
|
|
|
|
int nw=0,x=m_warnings.getlen();
|
|
|
|
if (!x || !display_warnings) return;
|
|
|
|
char *p=m_warnings.get();
|
|
|
|
while (x>0) if (!p[--x]) nw++;
|
|
|
|
fprintf(g_output,"\n%d warning%s:\n",nw,nw==1?"":"s");
|
|
|
|
for (x = 0; x < nw; x ++)
|
|
|
|
{
|
|
|
|
fprintf(g_output," %s\n",p);
|
|
|
|
p+=strlen(p)+1;
|
|
|
|
}
|
|
|
|
fflush(g_output);
|
|
|
|
}
|
2002-08-05 02:05:00 +00:00
|
|
|
|
2002-12-24 20:35:26 +00:00
|
|
|
void CEXEBuild::notify(int code, char *data)
|
|
|
|
{
|
|
|
|
if (notify_hwnd)
|
|
|
|
{
|
|
|
|
COPYDATASTRUCT cds = {(DWORD)code, strlen(data)+1, data};
|
|
|
|
SendMessage(notify_hwnd, WM_COPYDATA, 0, (LPARAM)&cds);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-08-05 02:05:00 +00:00
|
|
|
// Added by Ximon Eighteen 5th August 2002
|
|
|
|
#ifdef NSIS_CONFIG_PLUGIN_SUPPORT
|
2002-08-05 19:13:52 +00:00
|
|
|
void CEXEBuild::build_plugin_table(void)
|
2002-08-05 02:05:00 +00:00
|
|
|
{
|
2002-08-08 15:04:45 +00:00
|
|
|
plugin_used = false;
|
2002-08-29 10:25:45 +00:00
|
|
|
uninst_plugin_used = false;
|
2002-08-05 02:05:00 +00:00
|
|
|
char* nsisdir = definedlist.find("NSISDIR");
|
|
|
|
if (nsisdir)
|
|
|
|
{
|
2002-08-31 23:44:58 +00:00
|
|
|
char* searchPath = new char [strlen(nsisdir)+9];
|
2002-08-05 02:05:00 +00:00
|
|
|
if (searchPath)
|
|
|
|
{
|
2002-08-28 18:26:28 +00:00
|
|
|
wsprintf(searchPath,"%s\\plugins",nsisdir);
|
2002-08-05 19:13:52 +00:00
|
|
|
INFO_MSG("\nProcessing plugin dlls: \"%s\\*.dll\"\n",searchPath);
|
|
|
|
m_plugins.FindCommands(searchPath,display_info?true:false);
|
2002-08-05 02:05:00 +00:00
|
|
|
delete[] searchPath;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2002-08-09 22:12:10 +00:00
|
|
|
|
2003-05-24 13:50:24 +00:00
|
|
|
#define FLAG_OFFSET(flag) (FIELD_OFFSET(installer_flags, flag)/sizeof(int))
|
|
|
|
|
2002-09-21 13:11:28 +00:00
|
|
|
int CEXEBuild::add_plugins_dir_initializer(void)
|
2002-08-09 22:12:10 +00:00
|
|
|
{
|
2002-11-21 22:59:11 +00:00
|
|
|
if (!plugin_used && !uninst_plugin_used) return PS_OK;
|
2002-08-09 22:12:10 +00:00
|
|
|
|
2002-09-21 13:11:28 +00:00
|
|
|
SCRIPT_MSG("Adding plug-ins initializing function... ");
|
2002-08-09 22:12:10 +00:00
|
|
|
|
2002-11-21 22:59:11 +00:00
|
|
|
bool uninstall = !plugin_used;
|
2002-08-09 22:12:10 +00:00
|
|
|
|
2002-09-21 13:11:28 +00:00
|
|
|
int ret;
|
2002-08-09 22:12:10 +00:00
|
|
|
int zero_offset;
|
|
|
|
|
|
|
|
again:
|
2003-04-02 15:49:13 +00:00
|
|
|
// Function [un.]Initialize_____Plugins
|
2002-11-21 22:59:11 +00:00
|
|
|
ret=add_function(uninstall?"un.Initialize_____Plugins":"Initialize_____Plugins");
|
|
|
|
if (ret != PS_OK) return ret;
|
2002-10-01 17:16:49 +00:00
|
|
|
|
2003-04-02 15:49:13 +00:00
|
|
|
// don't move this, depends on [un.]
|
2002-08-09 22:12:10 +00:00
|
|
|
zero_offset=add_string("$0");
|
2003-04-02 15:49:13 +00:00
|
|
|
|
|
|
|
// SetDetailsPrint none
|
|
|
|
ret=add_entry_direct(EW_UPDATETEXT, 0, 4);
|
|
|
|
if (ret != PS_OK) return ret;
|
|
|
|
|
2002-08-09 22:12:10 +00:00
|
|
|
// StrCmp $PLUGINSDIR ""
|
2003-04-02 15:49:13 +00:00
|
|
|
ret=add_entry_direct(EW_STRCMP, add_string("$PLUGINSDIR"), 0, 0, ns_label.add("Initialize_____Plugins_done",0));
|
2002-08-09 22:12:10 +00:00
|
|
|
if (ret != PS_OK) return ret;
|
|
|
|
// Push $0
|
2003-04-02 15:49:13 +00:00
|
|
|
ret=add_entry_direct(EW_PUSHPOP, zero_offset);
|
|
|
|
if (ret != PS_OK) return ret;
|
|
|
|
// ClearErrors
|
2003-05-24 19:39:15 +00:00
|
|
|
ret=add_entry_direct(EW_SETFLAG, FLAG_OFFSET(exec_error));
|
2002-08-09 22:12:10 +00:00
|
|
|
if (ret != PS_OK) return ret;
|
2003-04-02 15:49:13 +00:00
|
|
|
// GetTempFileName $0
|
|
|
|
ret=add_entry_direct(EW_GETTEMPFILENAME);
|
2002-08-09 22:12:10 +00:00
|
|
|
if (ret != PS_OK) return ret;
|
2003-04-02 15:49:13 +00:00
|
|
|
// Delete $0 - the temp file created
|
|
|
|
ret=add_entry_direct(EW_DELETEFILE, zero_offset);
|
2002-08-09 22:12:10 +00:00
|
|
|
if (ret != PS_OK) return ret;
|
2003-04-02 15:49:13 +00:00
|
|
|
// CraeteDirectory $0 - a dir instead of that temp file
|
|
|
|
ret=add_entry_direct(EW_CREATEDIR, zero_offset);
|
2003-04-01 18:07:32 +00:00
|
|
|
if (ret != PS_OK) return ret;
|
2003-04-02 15:49:13 +00:00
|
|
|
// IfErrors Initialize_____Plugins_error - detect errors
|
2003-04-12 23:17:00 +00:00
|
|
|
ret=add_entry_direct(EW_IFFLAG, ns_label.add("Initialize_____Plugins_error",0), 0, FIELD_OFFSET(installer_flags, exec_error)/sizeof(int));
|
2003-04-01 18:07:32 +00:00
|
|
|
if (ret != PS_OK) return ret;
|
2002-08-09 22:12:10 +00:00
|
|
|
// Copy $0 to $PLUGINSDIR
|
2003-05-24 19:39:15 +00:00
|
|
|
ret=add_entry_direct(EW_ASSIGNVAR, 25, zero_offset);
|
2002-08-09 22:12:10 +00:00
|
|
|
if (ret != PS_OK) return ret;
|
|
|
|
// Pop $0
|
2003-04-02 15:49:13 +00:00
|
|
|
ret=add_entry_direct(EW_PUSHPOP, 0, 1);
|
2002-08-09 22:12:10 +00:00
|
|
|
if (ret != PS_OK) return ret;
|
|
|
|
|
2003-04-01 18:07:32 +00:00
|
|
|
// done
|
2002-08-09 22:12:10 +00:00
|
|
|
if (add_label("Initialize_____Plugins_done")) return PS_ERROR;
|
2003-04-02 15:49:13 +00:00
|
|
|
// Return
|
|
|
|
ret=add_entry_direct(EW_RET);
|
2003-04-01 18:07:32 +00:00
|
|
|
if (ret != PS_OK) return ret;
|
|
|
|
|
|
|
|
// error
|
|
|
|
if (add_label("Initialize_____Plugins_error")) return PS_ERROR;
|
|
|
|
// error message box
|
2003-04-02 15:49:13 +00:00
|
|
|
ret=add_entry_direct(EW_MESSAGEBOX, MB_OK|MB_ICONSTOP, add_string("Error! Can't initialize plug-ins directory. Please try again later."));
|
2003-04-01 18:07:32 +00:00
|
|
|
if (ret != PS_OK) return ret;
|
2003-04-02 15:49:13 +00:00
|
|
|
// Quit
|
|
|
|
ret=add_entry_direct(EW_QUIT);
|
2003-04-01 18:07:32 +00:00
|
|
|
if (ret != PS_OK) return ret;
|
2002-08-09 22:12:10 +00:00
|
|
|
|
2003-04-02 15:49:13 +00:00
|
|
|
// FunctionEnd
|
2002-08-09 22:12:10 +00:00
|
|
|
ret=function_end();
|
|
|
|
if (ret != PS_OK) return ret;
|
|
|
|
|
2002-08-29 10:25:45 +00:00
|
|
|
if (uninst_plugin_used && !uninstall) {
|
2002-08-09 22:12:10 +00:00
|
|
|
uninstall = true;
|
|
|
|
goto again;
|
|
|
|
}
|
|
|
|
|
|
|
|
SCRIPT_MSG("Done!\n");
|
|
|
|
|
|
|
|
return PS_OK;
|
|
|
|
}
|
2002-08-31 23:44:58 +00:00
|
|
|
#endif // NSIS_CONFIG_PLUGIN_SUPPORT
|
2002-11-09 13:51:40 +00:00
|
|
|
|
|
|
|
void CEXEBuild::init_res_editor()
|
|
|
|
{
|
|
|
|
build_compressor_set=true;
|
|
|
|
if (!res_editor)
|
|
|
|
res_editor=new CResourceEditor(header_data_new, exeheader_size_new);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CEXEBuild::close_res_editor()
|
|
|
|
{
|
|
|
|
if (!res_editor) return;
|
|
|
|
free(header_data_new);
|
|
|
|
header_data_new = res_editor->Save((DWORD&)exeheader_size_new);
|
|
|
|
delete res_editor;
|
|
|
|
res_editor=0;
|
2003-04-17 15:27:12 +00:00
|
|
|
}
|