summaryrefslogtreecommitdiff
path: root/plugins/Dbx_mmap_SA/Dbtool/src
diff options
context:
space:
mode:
authorVadim Dashevskiy <watcherhd@gmail.com>2012-07-23 07:10:03 +0000
committerVadim Dashevskiy <watcherhd@gmail.com>2012-07-23 07:10:03 +0000
commitd828acae837c71992f6a5f9dd688ba7afd944e56 (patch)
tree3be6506a6bbf40798199e07d4e2a0f0eef721efc /plugins/Dbx_mmap_SA/Dbtool/src
parent9126d2b133d00b836fca640f847a0948f7579e02 (diff)
Dbx_mmap_SA, Import_SA, Dbtool_SA: changed folder structure
git-svn-id: http://svn.miranda-ng.org/main/trunk@1108 1316c22d-e87f-b044-9b9b-93d7a3e3ba9c
Diffstat (limited to 'plugins/Dbx_mmap_SA/Dbtool/src')
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/Version.h9
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/aggressive.cpp59
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/cleaning.cpp63
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/contactchain.cpp105
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/database.h179
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/dbtool.h106
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/disk.cpp112
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/encryption.cpp299
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/eventchain.cpp363
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/fileaccess.cpp69
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/finaltasks.cpp78
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/finished.cpp82
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/initialchecks.cpp98
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/main.cpp33
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/modulechain.cpp146
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/openerror.cpp55
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/progress.cpp213
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/resource.h58
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/selectdb.cpp303
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/settingschain.cpp72
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/user.cpp71
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/utf.cpp69
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/welcome.cpp61
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/wizard.cpp137
-rw-r--r--plugins/Dbx_mmap_SA/Dbtool/src/worker.cpp76
25 files changed, 2916 insertions, 0 deletions
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/Version.h b/plugins/Dbx_mmap_SA/Dbtool/src/Version.h
new file mode 100644
index 0000000000..1ee2b8a4cf
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/Version.h
@@ -0,0 +1,9 @@
+#include "m_version.h"
+
+#define __FILEVERSION_STRING MIRANDA_VERSION_FILEVERSION
+#define __VERSION_STRING MIRANDA_VERSION_STRING
+#define __VERSION_DWORD MIRANDA_VERSION_DWORD
+#define __INTERNAL_NAME "dbtool"
+#define __FILENAME "dbtool.exe"
+#define __DESCRIPTION "Miranda IM Database Tool"
+#define __COPYRIGHT "© 2000-2011 Miranda IM project"
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/aggressive.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/aggressive.cpp
new file mode 100644
index 0000000000..5ec62707f1
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/aggressive.cpp
@@ -0,0 +1,59 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+#define BLOCKSIZE 65536
+
+extern DWORD sourceFileSize;
+extern DWORD spaceProcessed;
+extern DWORD sp;
+static DWORD ofsCurrent;
+
+int WorkAggressive(int firstTime)
+{
+ int blockBytes,i;
+ BYTE *buf;
+
+ if(firstTime) {
+ if (!opts.bAggressive) return ERROR_NO_MORE_ITEMS;
+ AddToStatus(STATUS_MESSAGE,TranslateT("Performing aggressive pass"));
+ ofsCurrent=0;
+ spaceProcessed=0;
+ sp=0;
+ }
+ blockBytes=min(BLOCKSIZE+3,(int)(sourceFileSize-ofsCurrent));
+ if(blockBytes<=0) return ERROR_NO_MORE_ITEMS;
+ buf = opts.pFile+ofsCurrent;
+ blockBytes-=3;
+ for(i=0;i<blockBytes;i++) {
+ if(buf[i]) {
+ if ((*(PDWORD)&buf[i]&0x00FFFFFF)!=0xDECADE)
+ AddToStatus(STATUS_WARNING,TranslateT("Aggressive: random junk at %08X: skipping"),ofsCurrent+i);
+ else {
+ //TODO: give user the option of placing manually
+ AddToStatus(STATUS_ERROR,TranslateT("Aggressive: unlinked data at %08X: can't automatically place"),ofsCurrent+i);
+ }
+ for (;i<blockBytes;i++)
+ if(buf[i]==0) {i--; break;}
+ }
+ }
+ ofsCurrent+=BLOCKSIZE;
+ spaceProcessed=ofsCurrent;
+ return ERROR_SUCCESS;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/cleaning.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/cleaning.cpp
new file mode 100644
index 0000000000..a16a11e7ac
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/cleaning.cpp
@@ -0,0 +1,63 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+INT_PTR CALLBACK CleaningDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam)
+{
+ INT_PTR bReturn;
+
+ if(DoMyControlProcessing(hdlg,message,wParam,lParam,&bReturn)) return bReturn;
+ switch(message) {
+ case WM_INITDIALOG:
+ CheckDlgButton(hdlg,IDC_ERASEHISTORY,opts.bEraseHistory);
+ EnableWindow(GetDlgItem(hdlg,IDC_ERASEHISTORY),!opts.bAggressive);
+ CheckDlgButton(hdlg,IDC_MARKREAD,opts.bMarkRead);
+ CheckDlgButton(hdlg,IDC_CONVERTUTF,opts.bConvertUtf);
+ TranslateDialogDefault(hdlg);
+ return TRUE;
+ case WZN_PAGECHANGING:
+ opts.bEraseHistory=IsDlgButtonChecked(hdlg,IDC_ERASEHISTORY)&&!opts.bAggressive;
+ opts.bMarkRead=IsDlgButtonChecked(hdlg,IDC_MARKREAD);
+ opts.bConvertUtf=IsDlgButtonChecked(hdlg,IDC_CONVERTUTF);
+ break;
+ case WM_COMMAND:
+ switch(LOWORD(wParam)) {
+ case IDC_BACK:
+ if(opts.bCheckOnly)
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_SELECTDB,(LPARAM)SelectDbDlgProc);
+ else
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_FILEACCESS,(LPARAM)FileAccessDlgProc);
+ break;
+ case IDOK:
+ if (!opts.hFile) {
+ opts.hFile = CreateFile( opts.filename, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL );
+ if ( opts.hFile == INVALID_HANDLE_VALUE ) {
+ opts.hFile = NULL;
+ opts.error = GetLastError();
+ SendMessage( GetParent(hdlg), WZM_GOTOPAGE, IDD_OPENERROR, ( LPARAM )OpenErrorDlgProc );
+ break;
+ }
+ }
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_PROGRESS,(LPARAM)ProgressDlgProc);
+ break;
+ }
+ break;
+ }
+ return FALSE;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/contactchain.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/contactchain.cpp
new file mode 100644
index 0000000000..0452f90d0d
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/contactchain.cpp
@@ -0,0 +1,105 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+int WorkSettingsChain(DWORD ofsContact,DBContact *dbc,int firstTime);
+int WorkEventChain(DWORD ofsContact,DBContact *dbc,int firstTime);
+
+static DWORD ofsThisContact,ofsDestPrevContact;
+static DWORD contactCount;
+static DWORD ofsDestThis,ofsNextContact;
+static int phase;
+static DBContact dbc;
+
+static void FinishUp(void)
+{
+ if(contactCount!=dbhdr.contactCount)
+ AddToStatus(STATUS_WARNING,TranslateT("Contact count marked wrongly: correcting"));
+ dbhdr.contactCount=contactCount;
+}
+
+int WorkContactChain(int firstTime)
+{
+ int first=0;
+
+ if(firstTime) {
+ AddToStatus(STATUS_MESSAGE,TranslateT("Processing contact chain"));
+ ofsDestPrevContact=0;
+ ofsThisContact=dbhdr.ofsFirstContact;
+ contactCount=0;
+ dbhdr.ofsFirstContact=0;
+ phase=0;
+ }
+
+ switch(phase) {
+ int ret;
+ case 0:
+ if(ofsThisContact==0) {
+ FinishUp();
+ return ERROR_NO_MORE_ITEMS;
+ }
+ if (!SignatureValid(ofsThisContact,DBCONTACT_SIGNATURE)) {
+ AddToStatus(STATUS_ERROR,TranslateT("Contact chain corrupted, further entries ignored"));
+ FinishUp();
+ return ERROR_NO_MORE_ITEMS;
+ }
+ if(ReadSegment(ofsThisContact,&dbc,sizeof(dbc))!=ERROR_SUCCESS) {
+ FinishUp();
+ return ERROR_NO_MORE_ITEMS;
+ }
+ ofsNextContact=dbc.ofsNext;
+ dbc.ofsNext=0;
+ if (!opts.bCheckOnly) {
+ if ((ofsDestThis=WriteSegment(WSOFS_END,&dbc,sizeof(dbc)))==WS_ERROR)
+ return ERROR_HANDLE_DISK_FULL;
+ if(ofsDestPrevContact)
+ WriteSegment(ofsDestPrevContact+offsetof(DBContact,ofsNext),&ofsDestThis,sizeof(DWORD));
+ else
+ dbhdr.ofsFirstContact=ofsDestThis;
+ } else
+ ofsDestThis = ofsThisContact; // needed in event chain worker
+ contactCount++;
+ phase++; first=1;
+ //fall thru
+ case 1:
+ ret=WorkSettingsChain(ofsDestThis,&dbc,first);
+ if(ret==ERROR_NO_MORE_ITEMS) {
+ phase++; first=1;
+ }
+ else if(ret) return ret;
+ else break;
+ //fall thru
+ case 2:
+ ret=WorkEventChain(ofsDestThis,&dbc,first);
+ if(ret==ERROR_NO_MORE_ITEMS) {
+ phase++; first=1;
+ }
+ else if(ret) return ret;
+ else break;
+ //fall thru
+ case 3:
+ if(WriteSegment(ofsDestThis,&dbc,sizeof(DBContact))==WS_ERROR)
+ return ERROR_HANDLE_DISK_FULL;
+ ofsDestPrevContact=ofsDestThis;
+ ofsThisContact=ofsNextContact;
+ phase=0;
+ break;
+ }
+ return ERROR_SUCCESS;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/database.h b/plugins/Dbx_mmap_SA/Dbtool/src/database.h
new file mode 100644
index 0000000000..aa73d16362
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/database.h
@@ -0,0 +1,179 @@
+/*
+
+Miranda IM: the free IM client for Microsoft* Windows*
+
+Copyright 2000-2005 Miranda ICQ/IM project,
+all portions of this codebase are copyrighted to the people
+listed in contributors.txt.
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+
+//all offsets are relative to the start of the file
+//offsets are 0 if there is nothing in the chain or this is the last in the
+//chain
+
+/* tree diagram
+
+DBHeader
+ |-->end of file (plain offset)
+ |-->first contact (DBContact)
+ | |-->next contact (DBContact)
+ | | \--> ...
+ | |-->first settings (DBContactSettings)
+ | | |-->next settings (DBContactSettings)
+ | | | \--> ...
+ | | \-->module name (DBModuleName)
+ | \-->first/last/firstunread event
+ |-->user contact (DBContact)
+ | |-->next contact=NULL
+ | |-->first settings as above
+ | \-->first/last/firstunread event as above
+ \-->first module name (DBModuleName)
+ \-->next module name (DBModuleName)
+ \--> ...
+*/
+
+#define DB_RESIZE_GRANULARITY 16384
+#define DB_THIS_VERSION 0x00000700u
+#define DB_SETTINGS_RESIZE_GRANULARITY 128
+
+#include <pshpack1.h>
+struct DBHeader {
+ BYTE signature[16]; // 'Miranda ICQ DB',0,26
+
+ WORD checkWord;
+ WORD cryptorUID;
+ DWORD ofsFileEnd; //offset of the end of the database - place to write
+ //new structures
+ DWORD slackSpace; //a counter of the number of bytes that have been
+ //wasted so far due to deleting structures and/or
+ //re-making them at the end. We should compact when
+ //this gets above a threshold
+ DWORD contactCount; //number of contacts in the chain,excluding the user
+ DWORD ofsFirstContact; //offset to first struct DBContact in the chain
+ DWORD ofsUser; //offset to struct DBContact representing the user
+ DWORD ofsFirstModuleName; //offset to first struct DBModuleName in the chain
+};
+
+#define DBCONTACT_SIGNATURE 0x43DECADEu
+struct DBContact {
+ DWORD signature;
+ DWORD ofsNext; //offset to the next contact in the chain. zero if
+ //this is the 'user' contact or the last contact
+ //in the chain
+ DWORD ofsFirstSettings; //offset to the first DBContactSettings in the
+ //chain for this contact.
+ DWORD eventCount; //number of events in the chain for this contact
+ DWORD ofsFirstEvent,ofsLastEvent; //offsets to the first and last DBEvent in
+ //the chain for this contact
+ DWORD ofsFirstUnreadEvent; //offset to the first (chronological) unread event
+ //in the chain, 0 if all are read
+ DWORD timestampFirstUnread; //timestamp of the event at ofsFirstUnreadEvent
+};
+
+#define DBMODULENAME_SIGNATURE 0x4DDECADEu
+struct DBModuleName {
+ DWORD signature;
+ DWORD ofsNext; //offset to the next module name in the chain
+ BYTE cbName; //number of characters in this module name
+ char name[1]; //name, no nul terminator
+};
+
+#define DBCONTACTSETTINGS_SIGNATURE 0x53DECADEu
+struct DBContactSettings {
+ DWORD signature;
+ DWORD ofsNext; //offset to the next contactsettings in the chain
+ DWORD ofsModuleName; //offset to the DBModuleName of the owner of these
+ //settings
+ DWORD cbBlob; //size of the blob in bytes. May be larger than the
+ //actual size for reducing the number of moves
+ //required using granularity in resizing
+ BYTE blob[1]; //the blob. a back-to-back sequence of DBSetting
+ //structs, the last has cbName=0
+};
+
+/* not a valid structure, content is figured out on the fly
+struct DBSetting {
+ BYTE cbName; //number of bytes in the name of this setting
+ //this =0 marks the end
+ char szName[...]; //setting name, excluding nul
+ BYTE dataType; //type of data. see m_database.h, db/contact/getsetting
+ union { //a load of types of data, length is defined by dataType
+ BYTE bVal; WORD wVal; DWORD dVal;
+ struct {
+ WORD cbString;
+ char szVal[...]; //excludes nul terminator
+ };
+ struct {
+ WORD cbBlob;
+ BYTE blobVal[...];
+ };
+ };
+};
+*/
+
+#define DBEVENT_SIGNATURE 0x45DECADEu
+struct DBEvent {
+ DWORD signature;
+ DWORD ofsPrev,ofsNext; //offset to the previous and next events in the
+ //chain. Chain is sorted chronologically
+ DWORD ofsModuleName; //offset to a DBModuleName struct of the name of
+ //the owner of this event
+ DWORD timestamp; //seconds since 00:00:00 01/01/1970
+ DWORD flags; //see m_database.h, db/event/add
+ WORD eventType; //module-defined event type
+ DWORD cbBlob; //number of bytes in the blob
+ BYTE blob[1]; //the blob. module-defined formatting
+};
+#include <poppack.h>
+
+//databasecorruption: called if any signatures are broken. very very fatal
+void DatabaseCorruption(const char *szPart);
+PBYTE DBRead(DWORD ofs,int bytesRequired,int *bytesAvail); //any preview result could be invalidated by the next call
+void DBWrite(DWORD ofs,PVOID pData,int count);
+void DBFlush(int setting);
+void DBMoveChunk(DWORD ofsDest,DWORD ofsSource,int bytes);
+DWORD CreateNewSpace(int bytes);
+void DeleteSpace(DWORD ofs,int bytes);
+void GetProfileDirectory(char *szPath,int cbPath);
+int GetDefaultProfilePath(char *szPath,int cbPath,int *specified);
+int ShouldShowProfileManager(void);
+
+#ifdef _DEBUG
+#define MAXCACHEDREADSIZE 512
+#else
+#define MAXCACHEDREADSIZE 2048 //technically 4096 would work, but I'm not going to push it
+#endif
+
+#ifdef _DEBUG
+//#define DBLOGGING
+#endif
+#ifdef DBLOGGING
+void DBLog(const char *file,int line,const char *fmt,...);
+#define logg() DBLog(__FILE__,__LINE__,"")
+#define log0(s) DBLog(__FILE__,__LINE__,s)
+#define log1(s,a) DBLog(__FILE__,__LINE__,s,a)
+#define log2(s,a,b) DBLog(__FILE__,__LINE__,s,a,b)
+#define log3(s,a,b,c) DBLog(__FILE__,__LINE__,s,a,b,c)
+#define log4(s,a,b,c,d) DBLog(__FILE__,__LINE__,s,a,b,c,d)
+#else
+#define logg()
+#define log0(s)
+#define log1(s,a)
+#define log2(s,a,b)
+#define log3(s,a,b,c)
+#define log4(s,a,b,c,d)
+#endif \ No newline at end of file
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/dbtool.h b/plugins/Dbx_mmap_SA/Dbtool/src/dbtool.h
new file mode 100644
index 0000000000..7f89ecd7aa
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/dbtool.h
@@ -0,0 +1,106 @@
+/*
+Miranda Database Tool
+Copyright 2000-2011 Miranda ICQ/IM project,
+all portions of this codebase are copyrighted to the people
+listed in contributors.txt.
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+
+#define _CRT_SECURE_NO_WARNINGS
+#define _CRT_NON_CONFORMING_SWPRINTFS
+
+#include <tchar.h>
+
+#include <windows.h>
+#include <stdio.h>
+#include <stddef.h>
+#include <io.h>
+#include <stdarg.h>
+#include <process.h>
+#include <direct.h>
+#include <malloc.h>
+#include <commctrl.h>
+#include <time.h>
+#include <win2k.h>
+
+#include "m_database.h"
+#include "database.h" // Note: This is a copy of database.h from the Miranda IM v0.3 tree.
+ // Remember to update this when releasing new dbtool versions.
+#include "resource.h"
+
+#define WZM_GOTOPAGE (WM_USER+1)
+#define WZN_PAGECHANGING (WM_USER+1221)
+#define WZN_CANCELCLICKED (WM_USER+1222)
+
+struct DbToolOptions {
+ TCHAR filename[MAX_PATH];
+ TCHAR workingFilename[MAX_PATH];
+ TCHAR outputFilename[MAX_PATH];
+ TCHAR backupFilename[MAX_PATH];
+ HANDLE hFile;
+ HANDLE hOutFile;
+ HANDLE hMap;
+ BYTE *pFile;
+ DWORD error;
+ int bCheckOnly,bBackup,bAggressive;
+ int bEraseHistory,bMarkRead,bConvertUtf;
+};
+
+extern HINSTANCE hInst;
+extern DbToolOptions opts;
+extern DBHeader dbhdr;
+
+int DoMyControlProcessing(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam,INT_PTR *bReturn);
+
+INT_PTR CALLBACK SelectDbDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam);
+INT_PTR CALLBACK CleaningDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam);
+INT_PTR CALLBACK ProgressDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam);
+INT_PTR CALLBACK FileAccessDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam);
+INT_PTR CALLBACK WizardDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam);
+INT_PTR CALLBACK FinishedDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam);
+INT_PTR CALLBACK WelcomeDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam);
+INT_PTR CALLBACK OpenErrorDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam);
+
+struct DBSignature {
+ char name[15];
+ BYTE eof;
+};
+static struct DBSignature dbSignature={"Miranda ICQ DB",0x1A};
+static struct DBSignature dbSignatureEncrypted={"Miranda ICQ SD",0x1A};
+
+#define SIZEOF(X) (sizeof(X)/sizeof(X[0]))
+
+#define STATUS_MESSAGE 0
+#define STATUS_WARNING 1
+#define STATUS_ERROR 2
+#define STATUS_FATAL 3
+#define STATUS_SUCCESS 4
+#define STATUS_CLASSMASK 0x0f
+int AddToStatus(DWORD flags, TCHAR* fmt,...);
+void SetProgressBar(int perThou);
+
+int PeekSegment(DWORD ofs,PVOID buf,int cbBytes);
+int ReadSegment(DWORD ofs,PVOID buf,int cbBytes);
+#define WSOFS_END 0xFFFFFFFF
+#define WS_ERROR 0xFFFFFFFF
+DWORD WriteSegment(DWORD ofs,PVOID buf,int cbBytes);
+int ReadWrittenSegment(DWORD ofs,PVOID buf,int cbBytes);
+int SignatureValid(DWORD ofs,DWORD signature);
+DWORD ConvertModuleNameOfs(DWORD ofsOld);
+void FreeModuleChain();
+extern BOOL bEncrypted;
+
+bool is_utf8_string(const char* str);
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/disk.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/disk.cpp
new file mode 100644
index 0000000000..1ac84a3dc6
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/disk.cpp
@@ -0,0 +1,112 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+extern DWORD spaceProcessed,sourceFileSize;
+
+int SignatureValid(DWORD ofs,DWORD signature)
+{
+ DWORD sig;
+
+ if(ofs+sizeof(sig)>=sourceFileSize) {
+ AddToStatus(STATUS_ERROR,TranslateT("Invalid offset found (database truncated?)"));
+ return 0;
+ }
+
+ sig = *(DWORD*)(opts.pFile+ofs);
+
+ return sig==signature;
+}
+
+int PeekSegment(DWORD ofs,PVOID buf,int cbBytes)
+{
+ DWORD bytesRead;
+
+ if(ofs>=sourceFileSize) {
+ AddToStatus(STATUS_ERROR,TranslateT("Invalid offset found"));
+ return ERROR_SEEK;
+ }
+
+ if (ofs+cbBytes>sourceFileSize)
+ bytesRead = sourceFileSize - ofs;
+ else
+ bytesRead = cbBytes;
+
+ if(bytesRead==0) {
+ AddToStatus(STATUS_ERROR,TranslateT("Error reading, database truncated? (%u)"),GetLastError());
+ return ERROR_READ_FAULT;
+ }
+
+ CopyMemory(buf, opts.pFile+ofs, bytesRead);
+
+ if ((int)bytesRead<cbBytes) return ERROR_HANDLE_EOF;
+ return ERROR_SUCCESS;
+}
+
+int ReadSegment(DWORD ofs,PVOID buf,int cbBytes)
+{
+ int ret;
+
+ ret=PeekSegment(ofs,buf,cbBytes);
+ if(ret!=ERROR_SUCCESS && ret!=ERROR_HANDLE_EOF) return ret;
+
+ if(opts.bAggressive) {
+ if (ofs+cbBytes>sourceFileSize) {
+ AddToStatus(STATUS_WARNING,TranslateT("Can't write to working file, aggressive mode may be too aggressive now"));
+ ZeroMemory(opts.pFile+ofs,sourceFileSize-ofs);
+ }
+ else
+ ZeroMemory(opts.pFile+ofs,cbBytes);
+ }
+ spaceProcessed+=cbBytes;
+ return ERROR_SUCCESS;
+}
+
+DWORD WriteSegment(DWORD ofs,PVOID buf,int cbBytes)
+{
+ DWORD bytesWritten;
+ if(opts.bCheckOnly) return 0xbfbfbfbf;
+ if(ofs==WSOFS_END) {
+ ofs=dbhdr.ofsFileEnd;
+ dbhdr.ofsFileEnd+=cbBytes;
+ }
+ SetFilePointer(opts.hOutFile,ofs,NULL,FILE_BEGIN);
+ WriteFile(opts.hOutFile,buf,cbBytes,&bytesWritten,NULL);
+ if ((int)bytesWritten<cbBytes) {
+ AddToStatus(STATUS_FATAL,TranslateT("Can't write to output file - disk full? (%u)"),GetLastError());
+ return WS_ERROR;
+ }
+ return ofs;
+}
+
+
+int ReadWrittenSegment(DWORD ofs,PVOID buf,int cbBytes)
+{
+ DWORD bytesRead;
+ if(opts.bCheckOnly) return 0xbfbfbfbf;
+ if(ofs + cbBytes > dbhdr.ofsFileEnd )
+ return ERROR_SEEK;
+
+ SetFilePointer(opts.hOutFile,ofs,NULL,FILE_BEGIN);
+ ReadFile(opts.hOutFile,buf,cbBytes,&bytesRead,NULL);
+ if ((int)bytesRead<cbBytes)
+ return ERROR_READ_FAULT;
+
+ return ERROR_SUCCESS;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/encryption.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/encryption.cpp
new file mode 100644
index 0000000000..1e7b264efa
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/encryption.cpp
@@ -0,0 +1,299 @@
+// (C) Artem Shpynov aka FYR and Igonin Vitaliy aka chaos.persei, 2007 - 2008
+
+#include "dbtool.h"
+
+typedef struct{
+ void* (__stdcall *GenerateKey)(char* pwd);
+ void (__stdcall *FreeKey)(void* key);
+ void (__stdcall *EncryptMem)(BYTE* data, int size, void* key);
+ void (__stdcall *DecryptMem)(BYTE* data, int size, void* key);
+
+ char* Name;
+ char* Info;
+ char* Author;
+ char* Site;
+ char* Email;
+
+ DWORD Version;
+
+ WORD uid;
+} Cryptor;
+
+typedef struct{
+ char dllname[255];
+ HMODULE hLib;
+ Cryptor* cryptor;
+} CryptoModule;
+
+BOOL CALLBACK DlgStdInProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam);
+
+char encryptKey[255];
+size_t encryptKeyLength;
+
+int wrongPass = 0;
+void* key;
+
+Cryptor* CryptoEngine = NULL;
+
+int ModulesCount = 0;
+CryptoModule* Modules[100];
+
+
+void zero_fill(BYTE * pBuf, size_t bufSize)
+{
+ size_t i;
+ for(i = 0; i < bufSize; i++)
+ pBuf[i] = 0;
+}
+
+void InitSecurity()
+{
+ HMODULE hLib;
+ WIN32_FIND_DATAA fd;
+
+ Cryptor* (__stdcall *GetCryptor)();
+
+ {
+ TCHAR szMirandaDir[MAX_PATH];
+ szMirandaDir[ 0 ] = 0;
+ TCHAR *str2;
+ GetModuleFileName(NULL,szMirandaDir,SIZEOF(szMirandaDir));
+ str2 = _tcsrchr(szMirandaDir,'\\');
+ if ( str2 != NULL )
+ *str2=0;
+ _tchdir(szMirandaDir);
+ }
+
+ HANDLE hFile = FindFirstFileA(".\\plugins\\cryptors\\*.dll", &fd);
+
+ AddToStatus(STATUS_MESSAGE,TranslateT("Scanning cryptors directory"));
+
+ ModulesCount = 0;
+ while (hFile != INVALID_HANDLE_VALUE)
+ {
+ char tmp[MAX_PATH], buf[255];
+ strcpy(tmp, ".\\plugins\\cryptors\\");
+ strcat(tmp, fd.cFileName);
+
+ hLib = LoadLibraryA(tmp);
+ if(hLib){
+ GetCryptor = (Cryptor* (__stdcall *)()) GetProcAddress(hLib, "GetCryptor");
+ if(GetCryptor){
+ TCHAR Name[100], Version[100], DllName[100];
+
+
+ Modules[ModulesCount] = (CryptoModule*) malloc(sizeof(CryptoModule));
+ Modules[ModulesCount]->cryptor = GetCryptor();
+ strcpy(Modules[ModulesCount]->dllname, fd.cFileName);
+ Modules[ModulesCount]->hLib = hLib;
+
+ _snprintf(buf,SIZEOF(buf),"%d.%d.%d.%d", HIBYTE(HIWORD(Modules[ModulesCount]->cryptor->Version)), LOBYTE(HIWORD(Modules[ModulesCount]->cryptor->Version)), HIBYTE(LOWORD(Modules[ModulesCount]->cryptor->Version)), LOBYTE(LOWORD(Modules[ModulesCount]->cryptor->Version)));
+
+ mbstowcs(Name, Modules[ModulesCount]->cryptor->Name, 100);
+ mbstowcs(Version, buf, 100);
+ mbstowcs(DllName, Modules[ModulesCount]->dllname, 100);
+
+ AddToStatus(STATUS_MESSAGE,TranslateT("Cryptor loaded: %s [%s] (%s)"), Name, Version, DllName);
+
+ ModulesCount++;
+ }else{
+ FreeLibrary(hLib);
+ }
+ }
+ if(ModulesCount >= 100) break;
+ if (!FindNextFileA(hFile, &fd)) break;
+ }
+
+ AddToStatus(STATUS_MESSAGE,TranslateT("%d crypto modules loaded"), ModulesCount);
+}
+
+void UnloadSecurity()
+{
+ int i;
+
+ if(CryptoEngine) CryptoEngine->FreeKey(key);
+
+ for(i = 0; i < ModulesCount; i++)
+ {
+ FreeLibrary(Modules[i]->hLib);
+ free(Modules[i]);
+ }
+}
+
+void EncoderInit()
+{
+ encryptKey[encryptKeyLength] = 0;
+ key = CryptoEngine->GenerateKey(encryptKey);
+}
+
+void EncodeCopyMemory(BYTE * dst, void * src, size_t size )
+{
+ memcpy(dst, src, size);
+ CryptoEngine->EncryptMem(dst, (int)size, key);
+}
+
+void DecodeCopyMemory(BYTE * dst, void * src, size_t size )
+{
+ memcpy(dst, src, size);
+ CryptoEngine->DecryptMem(dst, (int)size, key);
+}
+
+
+void EncodeMemory(BYTE * mem, size_t size)
+{
+ CryptoEngine->EncryptMem(mem, (int)size, key);
+}
+
+void DecodeMemory(BYTE * mem, size_t size)
+{
+ CryptoEngine->DecryptMem(mem, (int)size, key);
+}
+
+int bCheckingPass = 0;
+
+int CheckPassword(WORD checkWord, WORD cryptorUID, char * szDBName)
+{
+ WORD ver;
+ int res;
+
+ if(bCheckingPass) return 0;
+ bCheckingPass = 1;
+
+ {
+ int i;
+ int Found = 0;
+ for(i = 0; i < ModulesCount; i++) {
+ if(cryptorUID == Modules[i]->cryptor->uid){
+ CryptoEngine = Modules[i]->cryptor;
+ Found = 1;
+ break;
+ }
+ }
+ if (!Found){
+ AddToStatus(STATUS_FATAL, TranslateT("Sorry, but your database encrypted with unknown module"), MB_OK);
+ bCheckingPass = 0;
+ return 0;
+ }
+ }
+
+ {
+ wchar_t Name[100], Author[100];
+
+ mbstowcs(Name, CryptoEngine->Name, 100);
+ mbstowcs(Author, CryptoEngine->Author, 100);
+
+ AddToStatus(STATUS_MESSAGE, TranslateT("Database encrypted with %s by %s"), Name, Author);
+ }
+
+ while(1){
+ res = DialogBoxParam(hInst, MAKEINTRESOURCE(IDD_LOGIN), NULL, (DLGPROC)DlgStdInProc, (LPARAM)szDBName);
+ if(res == IDCANCEL)
+ {
+ wrongPass = 0;
+ bCheckingPass = 0;
+ return 0;
+ }
+ if(encryptKeyLength < 1) continue;
+ EncoderInit();
+ DecodeCopyMemory((BYTE*)&ver, &checkWord, sizeof(checkWord));
+ if(ver == 0x5195)
+ {
+ wrongPass = 0;
+ bCheckingPass = 0;
+ return 1;
+ }
+ wrongPass++;
+ }
+
+ bCheckingPass = 0;
+}
+
+
+UINT oldLangID = 0;
+void LanguageChanged(HWND hDlg)
+{
+ UINT LangID = (UINT)GetKeyboardLayout(0);
+ char Lang[3] = {0};
+ if (LangID != oldLangID)
+ {
+ oldLangID = LangID;
+ GetLocaleInfoA(MAKELCID((LangID & 0xffffffff), SORT_DEFAULT), LOCALE_SABBREVLANGNAME, Lang, 2);
+ SetDlgItemTextA(hDlg, IDC_LANG, Lang);
+ }
+}
+
+BOOL CALLBACK DlgStdInProc(HWND hDlg, UINT uMsg,WPARAM wParam,LPARAM lParam)
+{
+ switch(uMsg)
+ {
+ case WM_INITDIALOG:
+ {
+ HWND hwndCtrl;
+ TranslateDialogDefault(hDlg);
+
+ if(lParam && !wrongPass) SetDlgItemTextA(hDlg, IDC_DBNAME, (LPCSTR)lParam);
+ if(wrongPass)
+ {
+ if (wrongPass > 2)
+ {
+ hwndCtrl = GetDlgItem(hDlg, IDC_USERPASS);
+ EnableWindow(hwndCtrl, FALSE);
+ hwndCtrl = GetDlgItem(hDlg, IDOK);
+ EnableWindow(hwndCtrl, FALSE);
+
+ SetDlgItemText(hDlg, IDC_LOGININFO, TranslateT("Too many errors!"));
+
+ }
+ else
+ {
+ SetDlgItemText(hDlg, IDC_LOGININFO, TranslateT("Password is not correct!"));
+ }
+ }
+ oldLangID = 0;
+ SetTimer(hDlg,1,200,NULL);
+ LanguageChanged(hDlg);
+ return TRUE;
+ }
+
+ case WM_CTLCOLORSTATIC:
+ {
+ if ((HWND)lParam == GetDlgItem(hDlg, IDC_LANG))
+ {
+ SetTextColor((HDC)wParam, GetSysColor(COLOR_HIGHLIGHTTEXT));
+ SetBkMode((HDC)wParam, TRANSPARENT);
+ return (BOOL)GetSysColorBrush(COLOR_HIGHLIGHT);
+ }
+
+ return FALSE;
+ }
+
+ case WM_COMMAND:
+ {
+ UINT uid = LOWORD(wParam);
+
+ if(uid == IDOK){
+ if (!GetWindowLongPtr(hDlg,GWLP_USERDATA))
+ {
+ encryptKeyLength = GetDlgItemTextA(hDlg, IDC_USERPASS, encryptKey, 254);
+ EndDialog(hDlg,IDOK);
+ }else{
+
+ }
+ }else if(uid == IDCANCEL){
+ EndDialog(hDlg,IDCANCEL);
+ }
+ }
+ case WM_TIMER:
+ {
+ LanguageChanged(hDlg);
+ return FALSE;
+ }
+ case WM_DESTROY:
+ {
+ KillTimer(hDlg, 1);
+ return FALSE;
+ }
+ }
+
+ return FALSE;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/eventchain.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/eventchain.cpp
new file mode 100644
index 0000000000..3c9e37668a
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/eventchain.cpp
@@ -0,0 +1,363 @@
+/*
+Miranda Database Tool
+Miranda IM: the free IM client for Microsoft* Windows*
+
+Copyright 2000-2011 Miranda ICQ/IM project,
+all portions of this codebase are copyrighted to the people
+listed in contributors.txt.
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+static BOOL backLookup;
+static DWORD ofsThisEvent,ofsPrevEvent;
+static DWORD ofsDestPrevEvent;
+static DWORD eventCount;
+static DWORD lastTimestamp;
+static DWORD ofsFirstUnread,timestampFirstUnread;
+static DWORD memsize = 0;
+static DBEvent* memblock = NULL;
+static DBEvent* dbePrevEvent = NULL;
+
+static void ConvertOldEvent( DBEvent*& dbei )
+{
+ int msglen = (int)strlen(( char* )dbei->blob) + 1, msglenW = 0;
+ if ( msglen != (int) dbei->cbBlob ) {
+ int i, count = (( dbei->cbBlob - msglen ) / sizeof( WCHAR ));
+ WCHAR* p = ( WCHAR* )&dbei->blob[ msglen ];
+ for ( i=0; i < count; i++ ) {
+ if ( p[i] == 0 ) {
+ msglenW = i;
+ break;
+ } } }
+ else {
+ if ( !is_utf8_string(( char* )dbei->blob))
+ dbei->flags &= ~DBEF_UTF;
+ }
+
+ if ( msglenW > 0 && msglenW <= msglen ) {
+ char* utf8str = Utf8EncodeW(( WCHAR* )&dbei->blob[ msglen ] );
+ dbei->cbBlob = (DWORD)strlen( utf8str )+1;
+ dbei->flags |= DBEF_UTF;
+ if (offsetof(DBEvent,blob)+dbei->cbBlob > memsize) {
+ memsize = offsetof(DBEvent,blob)+dbei->cbBlob;
+ memblock = (DBEvent*)realloc(memblock, memsize);
+ dbei = memblock;
+ }
+ memcpy( &dbei->blob, utf8str, dbei->cbBlob );
+ free(utf8str);
+} }
+
+static void WriteOfsNextToPrevious(DWORD ofsPrev,DBContact *dbc,DWORD ofsNext)
+{
+ if(ofsPrev)
+ WriteSegment(ofsPrev+offsetof(DBEvent,ofsNext),&ofsNext,sizeof(DWORD));
+ else
+ dbc->ofsFirstEvent=ofsNext;
+}
+
+static void FinishUp(DWORD ofsLast,DBContact *dbc)
+{
+ WriteOfsNextToPrevious(ofsLast,dbc,0);
+ if(eventCount!=dbc->eventCount)
+ AddToStatus(STATUS_WARNING,TranslateT("Event count marked wrongly: correcting"));
+ dbc->eventCount=eventCount;
+ dbc->ofsLastEvent=ofsLast;
+ if(opts.bMarkRead) {
+ dbc->ofsFirstUnreadEvent=0;
+ dbc->timestampFirstUnread=0;
+ }
+ else {
+ dbc->ofsFirstUnreadEvent=ofsFirstUnread;
+ dbc->timestampFirstUnread=timestampFirstUnread;
+ }
+ if (memsize && memblock) {
+ free(memblock);
+ memsize = 0;
+ memblock = NULL;
+ }
+}
+
+static DWORD WriteEvent(DBEvent *dbe)
+{
+ DWORD ofs = WriteSegment( WSOFS_END, dbe, offsetof(DBEvent,blob)+dbe->cbBlob );
+ if ( ofs == WS_ERROR ) {
+ free( memblock );
+ memblock = NULL;
+ memsize = 0;
+ return 0;
+ }
+ return ofs;
+}
+
+int WorkEventChain(DWORD ofsContact,DBContact *dbc,int firstTime)
+{
+ DBEvent *dbeNew,dbeOld;
+ DBEvent *dbePrev = NULL;
+ DWORD ofsDestThis;
+ int isUnread=0;
+
+ if(firstTime) {
+ dbePrevEvent = NULL;
+ ofsPrevEvent=0;
+ ofsDestPrevEvent=0;
+ ofsThisEvent=dbc->ofsFirstEvent;
+ eventCount=0;
+ backLookup=0;
+ lastTimestamp=0;
+ ofsFirstUnread=timestampFirstUnread=0;
+ if(opts.bEraseHistory) {
+ dbc->eventCount=0;
+ dbc->ofsFirstEvent=0;
+ dbc->ofsLastEvent=0;
+ dbc->ofsFirstUnreadEvent=0;
+ dbc->timestampFirstUnread=0;
+ return ERROR_NO_MORE_ITEMS;
+ } }
+
+ if(ofsThisEvent==0) {
+ FinishUp(ofsDestPrevEvent,dbc);
+ return ERROR_NO_MORE_ITEMS;
+ }
+ if (!SignatureValid(ofsThisEvent,DBEVENT_SIGNATURE))
+ {
+ DWORD ofsNew = 0;
+ DWORD ofsTmp = dbc->ofsLastEvent;
+
+ if (!backLookup && ofsTmp) {
+ backLookup = 1;
+ while(SignatureValid(ofsTmp,DBEVENT_SIGNATURE))
+ {
+ if(PeekSegment(ofsTmp,&dbeOld,sizeof(dbeOld))!=ERROR_SUCCESS)
+ break;
+ ofsNew = ofsTmp;
+ ofsTmp = dbeOld.ofsPrev;
+ }
+ }
+ if (ofsNew) {
+ AddToStatus(STATUS_WARNING,TranslateT("Event chain corrupted, trying to recover..."));
+ ofsThisEvent = ofsNew;
+ } else {
+ AddToStatus(STATUS_ERROR,TranslateT("Event chain corrupted, further entries ignored"));
+ FinishUp(ofsDestPrevEvent,dbc);
+ return ERROR_NO_MORE_ITEMS;
+ }
+ }
+
+ if(PeekSegment(ofsThisEvent,&dbeOld,sizeof(dbeOld))!=ERROR_SUCCESS) {
+ FinishUp(ofsDestPrevEvent,dbc);
+ return ERROR_NO_MORE_ITEMS;
+ }
+
+ if(firstTime) {
+ if (!(dbeOld.flags&DBEF_FIRST)) {
+ AddToStatus(STATUS_WARNING,TranslateT("First event not marked as such: correcting"));
+ dbeOld.flags|=DBEF_FIRST;
+ }
+ dbeOld.ofsPrev=ofsContact;
+ lastTimestamp=dbeOld.timestamp;
+ }
+ else if(dbeOld.flags&DBEF_FIRST) {
+ AddToStatus(STATUS_WARNING,TranslateT("Event marked as first which is not: correcting"));
+ dbeOld.flags&=~DBEF_FIRST;
+ }
+
+ if(dbeOld.flags&~(DBEF_FIRST|DBEF_READ|DBEF_SENT|DBEF_RTL|DBEF_UTF)) {
+ AddToStatus(STATUS_WARNING,TranslateT("Extra flags found in event: removing"));
+ dbeOld.flags&=(DBEF_FIRST|DBEF_READ|DBEF_SENT|DBEF_RTL|DBEF_UTF);
+ }
+
+ if (!(dbeOld.flags&(DBEF_READ|DBEF_SENT))) {
+ if(opts.bMarkRead) dbeOld.flags|=DBEF_READ;
+ else if(ofsFirstUnread==0) {
+ if(dbc->ofsFirstUnreadEvent!=ofsThisEvent || dbc->timestampFirstUnread!=dbeOld.timestamp)
+ AddToStatus(STATUS_WARNING,TranslateT("First unread event marked wrong: fixing"));
+ isUnread=1;
+ } }
+
+ if(dbeOld.cbBlob>1024*1024 || dbeOld.cbBlob==0) {
+ AddToStatus(STATUS_ERROR,TranslateT("Infeasibly large event blob: skipping"));
+ ofsThisEvent=dbeOld.ofsNext;
+ return ERROR_SUCCESS;
+ }
+
+ if ( dbePrevEvent && dbeOld.timestamp == lastTimestamp ) {
+ int len = offsetof(DBEvent,blob)+dbePrevEvent->cbBlob;
+ dbePrev = (DBEvent*)malloc(len);
+ memcpy(dbePrev, dbePrevEvent, len);
+ }
+
+ if (offsetof(DBEvent,blob)+dbeOld.cbBlob > memsize) {
+ memsize = offsetof(DBEvent,blob)+dbeOld.cbBlob;
+ memblock = (DBEvent*)realloc(memblock, memsize);
+ }
+ dbeNew=memblock;
+
+ if(ReadSegment(ofsThisEvent,dbeNew,offsetof(DBEvent,blob)+dbeOld.cbBlob)!=ERROR_SUCCESS) {
+ FinishUp(ofsDestPrevEvent,dbc);
+ return ERROR_NO_MORE_ITEMS;
+ }
+
+ if ((dbeNew->ofsModuleName=ConvertModuleNameOfs(dbeOld.ofsModuleName))==0) {
+ ofsThisEvent=dbeOld.ofsNext;
+ return ERROR_SUCCESS;
+ }
+
+ if (!firstTime && dbeOld.ofsPrev!=ofsPrevEvent)
+ AddToStatus(STATUS_WARNING,TranslateT("Event not backlinked correctly: fixing"));
+
+ dbeNew->flags=dbeOld.flags;
+ dbeNew->ofsPrev=ofsDestPrevEvent;
+ dbeNew->ofsNext=0;
+
+ if ( dbeOld.eventType == EVENTTYPE_MESSAGE && opts.bConvertUtf )
+ ConvertOldEvent(dbeNew);
+
+ if ( dbePrev )
+ {
+ if ( dbePrev->cbBlob == dbeNew->cbBlob &&
+ dbePrev->ofsModuleName == dbeNew->ofsModuleName &&
+ dbePrev->eventType == dbeNew->eventType &&
+ (dbePrev->flags & DBEF_SENT) == (dbeNew->flags & DBEF_SENT) &&
+ !memcmp( dbePrev->blob, dbeNew->blob, dbeNew->cbBlob )
+ ) {
+ AddToStatus(STATUS_WARNING,TranslateT("Duplicate event was found: skipping"));
+ if (dbc->eventCount)
+ dbc->eventCount--;
+ free(dbePrev);
+ // ofsDestPrevEvent is still the same!
+ ofsPrevEvent=ofsThisEvent;
+ ofsThisEvent=dbeOld.ofsNext;
+ return ERROR_SUCCESS;
+ }
+ free(dbePrev);
+ }
+ else if ( !firstTime && dbeNew->timestamp < lastTimestamp )
+ {
+ DWORD found = 0;
+ DBEvent dbeTmp;
+ DWORD ofsTmp;
+
+ if (opts.bCheckOnly)
+ {
+ if (!opts.bAggressive)
+ {
+ ofsTmp = dbeOld.ofsPrev;
+ while(PeekSegment(ofsTmp,&dbeTmp,sizeof(dbeTmp))==ERROR_SUCCESS)
+ {
+ if (dbeTmp.ofsPrev == ofsContact) {
+ found = 1;
+ break;
+ }
+ if (dbeTmp.timestamp < dbeNew->timestamp) {
+ found = 2;
+ break;
+ }
+ ofsTmp = dbeTmp.ofsPrev;
+ }
+ }
+ AddToStatus(STATUS_WARNING,TranslateT("Event position in chain is not correct"));
+ }
+ else
+ {
+ ofsTmp = ofsDestPrevEvent;
+ while(ReadWrittenSegment(ofsTmp,&dbeTmp,sizeof(dbeTmp))==ERROR_SUCCESS)
+ {
+ if (dbeTmp.ofsPrev == ofsContact) {
+ found = 1;
+ break;
+ }
+ if (dbeTmp.timestamp < dbeNew->timestamp) {
+ found = 2;
+ break;
+ }
+ ofsTmp = dbeTmp.ofsPrev;
+ }
+ if (found)
+ AddToStatus(STATUS_WARNING,TranslateT("Event position in chain is not correct: fixing"));
+ else
+ AddToStatus(STATUS_WARNING,TranslateT("Event position in chain is not correct: unable to fix"));
+ }
+
+ // insert before FIRST
+ if (found == 1 && !opts.bCheckOnly) {
+ dbeNew->flags|=DBEF_FIRST;
+ dbeNew->ofsPrev=ofsContact;
+ dbeNew->ofsNext=dbc->ofsFirstEvent;
+
+ ofsDestThis = WriteEvent(dbeNew);
+ if ( !ofsDestThis )
+ return ERROR_HANDLE_DISK_FULL;
+
+ if ( isUnread && timestampFirstUnread >= dbeNew->timestamp ) {
+ ofsFirstUnread=ofsDestThis;
+ timestampFirstUnread=dbeNew->timestamp;
+ }
+ // fix first event
+ WriteOfsNextToPrevious(0,dbc,ofsDestThis);
+ // fix next event
+ WriteSegment(dbeNew->ofsNext+offsetof(DBEvent,ofsPrev),&ofsDestThis,sizeof(DWORD));
+ dbeTmp.flags &=~DBEF_FIRST;
+ WriteSegment(dbeNew->ofsNext+offsetof(DBEvent,flags),&dbeTmp.flags,sizeof(DWORD));
+ }
+ else if (found == 2 && !opts.bCheckOnly) {
+
+ dbeNew->ofsPrev=ofsTmp;
+ dbeNew->ofsNext=dbeTmp.ofsNext;
+
+ ofsDestThis = WriteEvent(dbeNew);
+ if ( !ofsDestThis )
+ return ERROR_HANDLE_DISK_FULL;
+
+ if ( isUnread && timestampFirstUnread >= dbeNew->timestamp ) {
+ ofsFirstUnread=ofsDestThis;
+ timestampFirstUnread=dbeNew->timestamp;
+ }
+ // fix previous event
+ WriteOfsNextToPrevious(dbeNew->ofsPrev,dbc,ofsDestThis);
+ // fix next event
+ WriteSegment(dbeNew->ofsNext+offsetof(DBEvent,ofsPrev),&ofsDestThis,sizeof(DWORD));
+ }
+
+ if (found) {
+ eventCount++;
+ // ofsDestPrevEvent is still the same!
+ ofsPrevEvent=ofsThisEvent;
+ ofsThisEvent=dbeOld.ofsNext;
+ return ERROR_SUCCESS;
+ }
+ }
+
+ lastTimestamp=dbeNew->timestamp;
+ dbePrevEvent = dbeNew;
+
+ ofsDestThis = WriteEvent(dbeNew);
+ if ( !ofsDestThis )
+ return ERROR_HANDLE_DISK_FULL;
+
+ if ( isUnread ) {
+ ofsFirstUnread=ofsDestThis;
+ timestampFirstUnread=dbeOld.timestamp;
+ }
+
+ eventCount++;
+ WriteOfsNextToPrevious(ofsDestPrevEvent,dbc,ofsDestThis);
+
+ ofsDestPrevEvent=ofsDestThis;
+ ofsPrevEvent=ofsThisEvent;
+ ofsThisEvent=dbeOld.ofsNext;
+ return ERROR_SUCCESS;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/fileaccess.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/fileaccess.cpp
new file mode 100644
index 0000000000..d904daf5b9
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/fileaccess.cpp
@@ -0,0 +1,69 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+INT_PTR CALLBACK FileAccessDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam)
+{
+ INT_PTR bReturn;
+
+ if(DoMyControlProcessing(hdlg,message,wParam,lParam,&bReturn)) return bReturn;
+ switch(message) {
+ case WM_INITDIALOG:
+ CheckDlgButton(hdlg,IDC_CHECKONLY,opts.bCheckOnly);
+ CheckDlgButton(hdlg,IDC_BACKUP,opts.bBackup);
+ CheckDlgButton(hdlg,IDC_AGGRESSIVE,opts.bAggressive);
+ SendMessage(hdlg,WM_COMMAND,MAKEWPARAM(IDC_CHECKONLY,BN_CLICKED),0);
+ TranslateDialogDefault(hdlg);
+ return TRUE;
+ case WZN_PAGECHANGING:
+ opts.bCheckOnly=IsDlgButtonChecked(hdlg,IDC_CHECKONLY);
+ opts.bAggressive=IsDlgButtonChecked(hdlg,IDC_AGGRESSIVE);
+ if(opts.bCheckOnly) opts.bBackup=0;
+ else opts.bBackup=IsDlgButtonChecked(hdlg,IDC_BACKUP);
+ break;
+ case WM_COMMAND:
+ switch(LOWORD(wParam)) {
+ case IDC_BACK:
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_SELECTDB,(LPARAM)SelectDbDlgProc);
+ break;
+ case IDOK:
+ if(opts.bCheckOnly) {
+ if (!opts.hFile) {
+ opts.hFile = CreateFile( opts.filename, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL );
+ if ( opts.hFile == INVALID_HANDLE_VALUE ) {
+ opts.hFile = NULL;
+ opts.error = GetLastError();
+ SendMessage( GetParent(hdlg), WZM_GOTOPAGE, IDD_OPENERROR, ( LPARAM )OpenErrorDlgProc );
+ break;
+ }
+ }
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_PROGRESS,(LPARAM)ProgressDlgProc);
+ }
+ else
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_CLEANING,(LPARAM)CleaningDlgProc);
+ break;
+ case IDC_CHECKONLY:
+ EnableWindow(GetDlgItem(hdlg,IDC_BACKUP),!IsDlgButtonChecked(hdlg,IDC_CHECKONLY));
+ EnableWindow(GetDlgItem(hdlg,IDC_STBACKUP),!IsDlgButtonChecked(hdlg,IDC_CHECKONLY));
+ break;
+ }
+ break;
+ }
+ return FALSE;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/finaltasks.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/finaltasks.cpp
new file mode 100644
index 0000000000..2ed8b7183d
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/finaltasks.cpp
@@ -0,0 +1,78 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+extern int errorCount;
+
+int WorkFinalTasks(int firstTime)
+{
+ FreeModuleChain();
+ AddToStatus(STATUS_MESSAGE,TranslateT("Processing final tasks"));
+ dbhdr.slackSpace=0;
+ if(WriteSegment(0,&dbhdr,sizeof(dbhdr))==WS_ERROR)
+ return ERROR_WRITE_FAULT;
+ if (opts.hFile) {
+ CloseHandle(opts.hFile);
+ opts.hFile = NULL;
+ }
+ if ( opts.hOutFile ) {
+ CloseHandle(opts.hOutFile);
+ opts.hOutFile = NULL;
+ }
+ if (opts.pFile) {
+ UnmapViewOfFile(opts.pFile);
+ opts.pFile = NULL;
+ }
+ if (opts.hMap) {
+ CloseHandle(opts.hMap);
+ opts.hMap = NULL;
+ }
+ if(errorCount && !opts.bBackup && !opts.bCheckOnly) {
+ extern time_t ts;
+ time_t dlg_ts = time(NULL);
+ if ( IDYES == MessageBox( NULL,
+ TranslateT("Errors were encountered, however you selected not to backup the original database. It is strongly recommended that you do so in case important data was omitted. Do you wish to keep a backup of the original database?"),
+ TranslateT("Miranda Database Tool"), MB_YESNO ))
+ opts.bBackup = 1;
+ ts += time(NULL) - dlg_ts;
+ }
+ if(opts.bBackup) {
+ int i;
+ TCHAR dbPath[MAX_PATH],dbFile[MAX_PATH];
+ _tcscpy( dbPath, opts.filename );
+ TCHAR* str2 = _tcsrchr( dbPath, '\\' );
+ if ( str2 != NULL ) {
+ _tcscpy( dbFile, str2+1 );
+ *str2 = 0;
+ }
+ else {
+ _tcscpy( dbFile, dbPath );
+ dbPath[0] = 0;
+ }
+ for(i=1;;i++) {
+ if(i==1) wsprintf(opts.backupFilename,TranslateT("%s\\Backup of %s"),dbPath,dbFile);
+ else wsprintf(opts.backupFilename,TranslateT("%s\\Backup (%d) of %s"),dbPath,i,dbFile);
+ if(_taccess(opts.backupFilename,0)==-1) break;
+ }
+ MoveFile(opts.filename,opts.backupFilename) || AddToStatus(STATUS_WARNING,TranslateT("Unable to rename original file"));
+ }
+ else if (!opts.bCheckOnly) DeleteFile(opts.filename) || AddToStatus(STATUS_WARNING,TranslateT("Unable to delete original file"));
+ if (!opts.bCheckOnly) MoveFile(opts.outputFilename,opts.filename) || AddToStatus(STATUS_WARNING,TranslateT("Unable to rename output file"));
+ return ERROR_NO_MORE_ITEMS;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/finished.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/finished.cpp
new file mode 100644
index 0000000000..08d0f5c893
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/finished.cpp
@@ -0,0 +1,82 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+#define WM_LAUNCHMIRANDA (WM_USER+1)
+INT_PTR CALLBACK FinishedDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam)
+{
+ INT_PTR bReturn;
+
+ if(DoMyControlProcessing(hdlg,message,wParam,lParam,&bReturn)) return bReturn;
+ switch(message) {
+ case WM_INITDIALOG:
+ EnableWindow(GetDlgItem(GetParent(hdlg),IDC_BACK),FALSE);
+ SetDlgItemText(GetParent(hdlg),IDCANCEL,TranslateT("&Finish"));
+ SetWindowLongPtr(GetDlgItem(hdlg,IDC_DBFILE),GWL_STYLE,GetWindowLongPtr(GetDlgItem(hdlg,IDC_DBFILE),GWL_STYLE)|SS_PATHELLIPSIS);
+ SetDlgItemText( hdlg, IDC_DBFILE, opts.filename );
+ if(opts.bBackup) {
+ ShowWindow(GetDlgItem(hdlg,IDC_STBACKUP),TRUE);
+ SetWindowLongPtr(GetDlgItem(hdlg,IDC_BACKUPFILE),GWL_STYLE,GetWindowLongPtr(GetDlgItem(hdlg,IDC_BACKUPFILE),GWL_STYLE)|SS_PATHELLIPSIS);
+ SetDlgItemText(hdlg,IDC_BACKUPFILE,opts.backupFilename);
+ }
+ else ShowWindow(GetDlgItem(hdlg,IDC_STBACKUP),FALSE);
+ TranslateDialogDefault(hdlg);
+ return TRUE;
+ case WM_LAUNCHMIRANDA:
+ if(IsDlgButtonChecked(hdlg,IDC_LAUNCHMIRANDA)) {
+ TCHAR dbFile[MAX_PATH],dbPath[MAX_PATH],*str2;
+ _tcscpy(dbPath,opts.filename);
+ str2 = _tcsrchr(dbPath,'\\');
+ if ( str2 == NULL ) {
+ _tcscpy( dbFile, dbPath );
+ dbPath[ 0 ] = 0;
+ }
+ else {
+ _tcscpy( dbFile, str2+1 );
+ *str2 = 0;
+ }
+ str2 = _tcsrchr( dbFile, '.' );
+ if ( str2 != NULL )
+ *str2 = 0;
+ _tcscat( dbPath, _T("\\miranda32.exe"));
+ if ( GetFileAttributes( dbPath ) == INVALID_FILE_ATTRIBUTES ) {
+ GetModuleFileName( NULL, dbPath, SIZEOF( dbPath ));
+ if (( str2 = _tcsrchr( dbPath, '\\' )) != NULL )
+ *str2 = 0;
+ else
+ dbPath[0] = 0;
+ _tcscat( dbPath, _T("\\miranda32.exe"));
+ }
+ ShellExecute( hdlg, NULL, dbPath, dbFile, _T(""), SW_SHOW );
+ }
+ break;
+ case WZN_CANCELCLICKED:
+ SendMessage(hdlg,WM_LAUNCHMIRANDA,0,0);
+ break;
+ case WM_COMMAND:
+ switch(LOWORD(wParam)) {
+ case IDOK:
+ SendMessage(hdlg,WM_LAUNCHMIRANDA,0,0);
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_SELECTDB,(LPARAM)SelectDbDlgProc);
+ break;
+ }
+ break;
+ }
+ return FALSE;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/initialchecks.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/initialchecks.cpp
new file mode 100644
index 0000000000..385d31cb92
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/initialchecks.cpp
@@ -0,0 +1,98 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+BOOL bEncrypted;
+extern DWORD sourceFileSize,spaceUsed;
+int CheckPassword(WORD version, WORD cryptorUID, char * szDBName);
+void InitSecurity();
+
+int WorkInitialChecks(int firstTime)
+{
+ DWORD bytesRead;
+
+ sourceFileSize=GetFileSize(opts.hFile,NULL);
+ if(sourceFileSize==0) {
+ AddToStatus(STATUS_WARNING,TranslateT("Database is newly created and has no data to process"));
+ AddToStatus(STATUS_SUCCESS,TranslateT("Processing completed successfully"));
+ return ERROR_INVALID_DATA;
+ }
+ ReadFile(opts.hFile,&dbhdr,sizeof(dbhdr),&bytesRead,NULL);
+ if(bytesRead<sizeof(dbhdr)) {
+ AddToStatus(STATUS_FATAL,TranslateT("Database is corrupted and too small to contain any recoverable data"));
+ return ERROR_BAD_FORMAT;
+ }
+ bEncrypted = false;
+ if(memcmp(dbhdr.signature,&dbSignature,sizeof(dbhdr.signature))) {
+ if(memcmp(dbhdr.signature,&dbSignatureEncrypted,sizeof(dbhdr.signature))) {
+ AddToStatus(STATUS_FATAL,TranslateT("Database signature is corrupted, automatic repair is impossible"));
+ return ERROR_BAD_FORMAT;
+ }
+ AddToStatus(STATUS_SUCCESS,TranslateT("Database is Secured MMAP database"));
+ InitSecurity();
+ if(CheckPassword(dbhdr.checkWord, dbhdr.cryptorUID, (char*)Translate("your database"))) {
+ AddToStatus(STATUS_SUCCESS,TranslateT("Secured MMAP: authorization successful"));
+ bEncrypted = true;
+ }
+ else
+ {
+ AddToStatus(STATUS_FATAL,TranslateT("You are not authorized for access to Database"));
+ return ERROR_BAD_FORMAT;
+ }
+ }
+ if(dbhdr.checkWord!=0x0700 && !bEncrypted) {
+ AddToStatus(STATUS_FATAL,TranslateT("Database is marked as belonging to an unknown version of Miranda"));
+ return ERROR_BAD_FORMAT;
+ }
+ _tcscpy(opts.workingFilename,opts.filename);
+
+ if(opts.bCheckOnly) {
+ _tcscpy( opts.outputFilename, TranslateT("<check only>"));
+ opts.hOutFile=INVALID_HANDLE_VALUE;
+ }
+ else {
+ _tcscpy(opts.outputFilename,opts.filename);
+ *_tcsrchr(opts.outputFilename,'.')=0;
+ _tcscat(opts.outputFilename,TranslateT(" (Output).dat"));
+ opts.hOutFile = CreateFile(opts.outputFilename,GENERIC_WRITE,FILE_SHARE_READ,NULL,CREATE_ALWAYS,FILE_FLAG_SEQUENTIAL_SCAN,NULL);
+ if ( opts.hOutFile == INVALID_HANDLE_VALUE ) {
+ AddToStatus(STATUS_FATAL,TranslateT("Can't create output file (%u)"),GetLastError());
+ return ERROR_ACCESS_DENIED;
+ }
+ }
+
+ opts.hMap = CreateFileMapping(opts.hFile, NULL, opts.bAggressive?PAGE_WRITECOPY:PAGE_READONLY, 0, 0, NULL);
+
+ if (opts.hMap)
+ opts.pFile = (BYTE*)MapViewOfFile(opts.hMap, opts.bAggressive?FILE_MAP_COPY:FILE_MAP_READ, 0, 0 ,0);
+ else {
+ AddToStatus(STATUS_FATAL,TranslateT("Can't create file mapping (%u)"),GetLastError());
+ return ERROR_ACCESS_DENIED;
+ }
+
+ if (!opts.pFile) {
+ AddToStatus(STATUS_FATAL,TranslateT("Can't create map view of file (%u)"),GetLastError());
+ return ERROR_ACCESS_DENIED;
+ }
+ if(ReadSegment(0,&dbhdr,sizeof(dbhdr))!=ERROR_SUCCESS) return ERROR_READ_FAULT;
+ if(WriteSegment(0,&dbhdr,sizeof(dbhdr))==WS_ERROR) return ERROR_HANDLE_DISK_FULL;
+ spaceUsed=dbhdr.ofsFileEnd-dbhdr.slackSpace;
+ dbhdr.ofsFileEnd=sizeof(dbhdr);
+ return ERROR_NO_MORE_ITEMS;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/main.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/main.cpp
new file mode 100644
index 0000000000..d505751bd5
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/main.cpp
@@ -0,0 +1,33 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+HINSTANCE hInst;
+int hLangpack=0;
+
+DbToolOptions opts = {0};
+
+int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
+{
+ hInst=hInstance;
+ InitCommonControls();
+ LoadLangPackModule();
+ DialogBox(hInst,MAKEINTRESOURCE(IDD_WIZARD),NULL,WizardDlgProc);
+ return 0;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/modulechain.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/modulechain.cpp
new file mode 100644
index 0000000000..2b91059176
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/modulechain.cpp
@@ -0,0 +1,146 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+struct ModChainEntry {
+ DWORD ofsOld,ofsNew;
+ int size;
+ char name[257];
+} static *modChain=NULL;
+static int modChainCount;
+static DWORD ofsCurrent;
+static int phase,iCurrentModName;
+static DWORD ofsLast;
+static int last_mod = 0;
+
+int WorkModuleChain(int firstTime)
+{
+ DBModuleName moduleName,*newModName;
+
+ if(firstTime) {
+ AddToStatus(STATUS_MESSAGE,TranslateT("Processing module name chain"));
+ modChainCount=0;
+ last_mod = 0;
+ if(modChain!=NULL) free(modChain);
+ modChain = (ModChainEntry*)malloc(sizeof(ModChainEntry));
+ phase=0;
+ ofsCurrent=dbhdr.ofsFirstModuleName;
+ }
+ switch(phase) {
+ case 0:
+ if(ofsCurrent==0) {
+ phase++;
+ return ERROR_SUCCESS;
+ }
+ if (!SignatureValid(ofsCurrent,DBMODULENAME_SIGNATURE)) {
+ AddToStatus(STATUS_ERROR,TranslateT("Module chain corrupted, further entries ignored"));
+ phase++;
+ return ERROR_SUCCESS;
+ }
+ if(PeekSegment(ofsCurrent,&moduleName,offsetof(DBModuleName,name))!=ERROR_SUCCESS) {
+ phase++;
+ return ERROR_SUCCESS;
+ }
+ if(moduleName.cbName>256)
+ AddToStatus(STATUS_WARNING,TranslateT("Unreasonably long module name, skipping"));
+ else {
+ modChain=(ModChainEntry*)realloc(modChain,sizeof(ModChainEntry)*++modChainCount);
+
+ modChain[modChainCount-1].ofsOld=ofsCurrent;
+ modChain[modChainCount-1].size=offsetof(DBModuleName,name)+moduleName.cbName;
+ modChain[modChainCount-1].ofsNew=0;
+
+ if (moduleName.cbName)
+ PeekSegment(ofsCurrent+offsetof(DBModuleName,name),&modChain[modChainCount-1].name,moduleName.cbName);
+ modChain[modChainCount-1].name[moduleName.cbName]=0;
+ }
+ ofsCurrent=moduleName.ofsNext;
+ break;
+ case 1:
+ ofsLast = 0;
+ iCurrentModName=0;
+ dbhdr.ofsFirstModuleName=0;
+ phase++;
+ case 2:
+ if(iCurrentModName>=modChainCount) {
+ DWORD dw = 0;
+ if(ofsLast) WriteSegment(ofsLast+offsetof(DBModuleName,ofsNext),&dw,sizeof(DWORD));
+ return ERROR_NO_MORE_ITEMS;
+ }
+ if(modChain[iCurrentModName].ofsNew==0) {
+ newModName=(DBModuleName*)_alloca(modChain[iCurrentModName].size);
+ if(ReadSegment(modChain[iCurrentModName].ofsOld,newModName,modChain[iCurrentModName].size)!=ERROR_SUCCESS)
+ return ERROR_NO_MORE_ITEMS;
+ if ((modChain[iCurrentModName].ofsNew=WriteSegment(WSOFS_END,newModName,modChain[iCurrentModName].size))==WS_ERROR)
+ return ERROR_HANDLE_DISK_FULL;
+ { // check duplicated modulenames
+ int i, n=0;
+ for(i=iCurrentModName+1;i<modChainCount;i++)
+ if (!strcmp(modChain[i].name, modChain[iCurrentModName].name)) {
+ modChain[i].ofsNew = modChain[iCurrentModName].ofsNew;
+ n++;
+ }
+ if (n) {
+ TCHAR *pszModuleName;
+
+ TCHAR szModuleName[257];
+ MultiByteToWideChar(CP_ACP, 0, modChain[iCurrentModName].name, -1, szModuleName, sizeof(szModuleName) / sizeof(TCHAR));
+ pszModuleName = szModuleName;
+
+ AddToStatus(STATUS_WARNING,TranslateT("Module name '%s' is not unique: %d duplicates found)"), pszModuleName, n);
+ }
+ }
+ if(iCurrentModName==0)
+ dbhdr.ofsFirstModuleName=modChain[iCurrentModName].ofsNew;
+ else
+ if(WriteSegment(ofsLast+offsetof(DBModuleName,ofsNext),&modChain[iCurrentModName].ofsNew,sizeof(DWORD))==WS_ERROR)
+ return ERROR_HANDLE_DISK_FULL;
+ ofsLast = modChain[iCurrentModName].ofsNew;
+ }
+ iCurrentModName++;
+ break;
+ }
+ return ERROR_SUCCESS;
+}
+
+DWORD ConvertModuleNameOfs(DWORD ofsOld)
+{
+ int i;
+
+ if ( modChain[last_mod].ofsOld==ofsOld )
+ return modChain[last_mod].ofsNew;
+
+ for(i=0;i<modChainCount;i++)
+ if(modChain[i].ofsOld==ofsOld) {
+ last_mod = i;
+ return modChain[last_mod].ofsNew;
+ }
+
+ AddToStatus(STATUS_ERROR,TranslateT("Invalid module name offset, skipping data"));
+ return 0;
+}
+
+void FreeModuleChain()
+{
+ if(modChain!=NULL) {
+ free(modChain);
+ modChain = NULL;
+ last_mod = 0;
+ }
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/openerror.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/openerror.cpp
new file mode 100644
index 0000000000..1fa2964c20
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/openerror.cpp
@@ -0,0 +1,55 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+INT_PTR CALLBACK OpenErrorDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam)
+{
+ INT_PTR bReturn;
+
+ if(DoMyControlProcessing(hdlg,message,wParam,lParam,&bReturn)) return bReturn;
+ switch(message) {
+ case WM_INITDIALOG:
+ { TCHAR szError[256];
+ FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM,NULL,opts.error,0,szError,SIZEOF(szError),NULL);
+ SetDlgItemText(hdlg,IDC_ERRORTEXT,szError);
+ }
+ if(opts.error==ERROR_SHARING_VIOLATION) ShowWindow(GetDlgItem(hdlg,IDC_INUSE),SW_SHOW);
+ SetWindowLongPtr( GetDlgItem( hdlg, IDC_FILE ), GWL_STYLE, GetWindowLongPtr( GetDlgItem( hdlg, IDC_FILE ), GWL_STYLE ) | SS_PATHELLIPSIS );
+ TranslateDialogDefault( hdlg );
+ SetDlgItemText( hdlg, IDC_FILE, opts.filename );
+ return TRUE;
+ case WM_COMMAND:
+ switch(LOWORD(wParam)) {
+ case IDC_BACK:
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_SELECTDB,(LPARAM)SelectDbDlgProc);
+ break;
+ case IDOK:
+ opts.hFile = CreateFile( opts.filename, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL );
+ if ( opts.hFile == INVALID_HANDLE_VALUE ) {
+ opts.hFile = NULL;
+ opts.error = GetLastError();
+ SendMessage( GetParent( hdlg ), WZM_GOTOPAGE, IDD_OPENERROR, ( LPARAM )OpenErrorDlgProc );
+ }
+ else SendMessage( GetParent( hdlg ), WZM_GOTOPAGE, IDD_FILEACCESS, ( LPARAM )FileAccessDlgProc );
+ break;
+ }
+ break;
+ }
+ return FALSE;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/progress.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/progress.cpp
new file mode 100644
index 0000000000..5ddb8bd985
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/progress.cpp
@@ -0,0 +1,213 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+#define WM_PROCESSINGDONE (WM_USER+1)
+
+void __cdecl WorkerThread(void *unused);
+static HWND hwndStatus,hdlgProgress,hwndBar;
+HANDLE hEventRun = NULL, hEventAbort = NULL;
+int errorCount;
+
+int AddToStatus(DWORD flags, TCHAR* fmt,...)
+{
+ va_list vararg;
+ TCHAR str[256];
+ int i,ret;
+
+ va_start(vararg,fmt);
+ ret=_vsntprintf(str,SIZEOF(str),fmt,vararg);
+ va_end(vararg);
+ i=SendMessage(hwndStatus,LB_ADDSTRING,0,(LPARAM)str);
+ SendMessage(hwndStatus,LB_SETITEMDATA,i,flags);
+ InvalidateRect(hwndStatus,NULL,FALSE);
+ SendMessage(hwndStatus,LB_SETTOPINDEX,i,0);
+#ifdef _DEBUG
+ OutputDebugString(str);
+ OutputDebugStringA("\n");
+#endif
+ if ((flags&STATUS_CLASSMASK)==STATUS_ERROR) errorCount++;
+ return ret;
+}
+
+void SetProgressBar(int perThou)
+{
+ SendMessage(hwndBar,PBM_SETPOS,perThou,0);
+}
+
+void ProcessingDone(void)
+{
+ if (opts.pFile) {
+ UnmapViewOfFile(opts.pFile);
+ opts.pFile = NULL;
+ }
+ if (opts.hMap) {
+ CloseHandle(opts.hMap);
+ opts.hMap = NULL;
+ }
+ SendMessage(hdlgProgress,WM_PROCESSINGDONE,0,0);
+}
+
+INT_PTR CALLBACK ProgressDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam)
+{
+ INT_PTR bReturn;
+ static int fontHeight,listWidth;
+ static int manualAbort;
+ static HFONT hBoldFont = NULL;
+
+ if(DoMyControlProcessing(hdlg,message,wParam,lParam,&bReturn)) return bReturn;
+ switch(message) {
+ case WM_INITDIALOG:
+ EnableWindow(GetDlgItem(GetParent(hdlg),IDOK),FALSE);
+ hdlgProgress=hdlg;
+ hwndStatus=GetDlgItem(hdlg,IDC_STATUS);
+ errorCount=0;
+ hwndBar=GetDlgItem(hdlg,IDC_PROGRESS);
+ SendMessage(hwndBar,PBM_SETRANGE,0,MAKELPARAM(0,1000));
+ { HDC hdc;
+ HFONT hFont,hoFont;
+ SIZE s;
+ hdc=GetDC(NULL);
+ hFont=(HFONT)SendMessage(hdlg,WM_GETFONT,0,0);
+ hoFont=(HFONT)SelectObject(hdc,hFont);
+ GetTextExtentPoint32(hdc,_T("x"),1,&s);
+ SelectObject(hdc,hoFont);
+ ReleaseDC(NULL,hdc);
+ fontHeight=s.cy;
+ }
+ { RECT rc;
+ GetClientRect(GetDlgItem(hdlg,IDC_STATUS),&rc);
+ listWidth=rc.right;
+ }
+ {
+ LOGFONT lf;
+ GetObject((HFONT)SendDlgItemMessage(hdlg,IDC_STATUS,WM_GETFONT,0,0),sizeof(lf),&lf);
+ lf.lfWeight=FW_BOLD;
+ hBoldFont=CreateFontIndirect(&lf);
+ }
+ manualAbort=0;
+ hEventRun=CreateEvent(NULL,TRUE,TRUE,NULL);
+ hEventAbort=CreateEvent(NULL,TRUE,FALSE,NULL);
+ TranslateDialogDefault(hdlg);
+ _beginthread(WorkerThread,0,NULL);
+ return TRUE;
+ case WM_MEASUREITEM:
+ { LPMEASUREITEMSTRUCT mis=(LPMEASUREITEMSTRUCT)lParam;
+ mis->itemWidth=listWidth;
+ mis->itemHeight=fontHeight;
+ return TRUE;
+ }
+ case WM_DRAWITEM:
+ { LPDRAWITEMSTRUCT dis=(LPDRAWITEMSTRUCT)lParam;
+ TCHAR str[256];
+ int bold=0;
+ HFONT hoFont;
+ if ((int)dis->itemID==-1) break;
+ SendMessage(dis->hwndItem,LB_GETTEXT,dis->itemID,(LPARAM)str);
+ switch(dis->itemData&STATUS_CLASSMASK) {
+ case STATUS_MESSAGE:
+ SetTextColor(dis->hDC,RGB(0,0,0));
+ break;
+ case STATUS_WARNING:
+ SetTextColor(dis->hDC,RGB(192,128,0));
+ break;
+ case STATUS_ERROR:
+ SetTextColor(dis->hDC,RGB(192,0,0));
+ break;
+ case STATUS_FATAL:
+ bold=1;
+ SetTextColor(dis->hDC,RGB(192,0,0));
+ break;
+ case STATUS_SUCCESS:
+ bold=1;
+ SetTextColor(dis->hDC,RGB(0,192,0));
+ break;
+ }
+ if(bold) hoFont=(HFONT)SelectObject(dis->hDC,hBoldFont);
+ ExtTextOut(dis->hDC,dis->rcItem.left,dis->rcItem.top,ETO_CLIPPED|ETO_OPAQUE,&dis->rcItem,str,(UINT)_tcslen(str),NULL);
+ if(bold) SelectObject(dis->hDC,hoFont);
+ return TRUE;
+ }
+ case WM_PROCESSINGDONE:
+ SetProgressBar(1000);
+ EnableWindow(GetDlgItem(GetParent(hdlg),IDOK),TRUE);
+ if(manualAbort==1)
+ EndDialog(GetParent(hdlg),0);
+ else if(manualAbort==2) {
+ if(opts.bCheckOnly)
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_FILEACCESS,(LPARAM)FileAccessDlgProc);
+ else {
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_CLEANING,(LPARAM)CleaningDlgProc);
+ CloseHandle(opts.hOutFile);
+ opts.hOutFile = NULL;
+ }
+ break;
+ }
+ AddToStatus(STATUS_SUCCESS,TranslateT("Click Next to continue"));
+ break;
+ case WZN_CANCELCLICKED:
+ ResetEvent(hEventRun);
+ if(IsWindowEnabled(GetDlgItem(GetParent(hdlg),IDOK))) break;
+ if(MessageBox(hdlg,TranslateT("Processing has not yet completed, if you cancel now then the changes that have currently been made will be rolled back and the original database will be restored. Do you still want to cancel?"),TranslateT("Miranda Database Tool"),MB_YESNO)==IDYES) {
+ manualAbort=1;
+ SetEvent(hEventAbort);
+ }
+ SetEvent(hEventRun);
+ SetWindowLongPtr(hdlg,DWLP_MSGRESULT,TRUE);
+ return TRUE;
+ case WM_COMMAND:
+ switch(LOWORD(wParam)) {
+ case IDC_BACK:
+ ResetEvent(hEventRun);
+ if (!IsWindowEnabled(GetDlgItem(GetParent(hdlg),IDOK))) {
+ if(MessageBox(hdlg,TranslateT("Processing has not yet completed, if you go back now then the changes that have currently been made will be rolled back and the original database will be restored. Do you still want to go back?"),TranslateT("Miranda Database Tool"),MB_YESNO)==IDYES) {
+ manualAbort=2;
+ SetEvent(hEventAbort);
+ }
+ SetEvent(hEventRun);
+ break;
+ }
+ SetEvent(hEventRun);
+ if(opts.bCheckOnly)
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_FILEACCESS,(LPARAM)FileAccessDlgProc);
+ else
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_CLEANING,(LPARAM)CleaningDlgProc);
+ break;
+ case IDOK:
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_FINISHED,(LPARAM)FinishedDlgProc);
+ break;
+ }
+ break;
+ case WM_DESTROY:
+ if (hEventAbort) {
+ CloseHandle(hEventAbort);
+ hEventAbort = NULL;
+ }
+ if (hEventRun) {
+ CloseHandle(hEventRun);
+ hEventRun = NULL;
+ }
+ if (hBoldFont) {
+ DeleteObject(hBoldFont);
+ hBoldFont = NULL;
+ }
+ break;
+ }
+ return FALSE;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/resource.h b/plugins/Dbx_mmap_SA/Dbtool/src/resource.h
new file mode 100644
index 0000000000..f92176c33f
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/resource.h
@@ -0,0 +1,58 @@
+//{{NO_DEPENDENCIES}}
+// Microsoft Visual C++ generated include file.
+// Used by resource.rc
+//
+#define IDC_BACK 3
+#define IDD_WIZARD 101
+#define IDI_DBTOOL 102
+#define IDI_PROFILEGREEN 104
+#define IDR_DEFAULT1 104
+#define IDI_PROFILEYELLOW 105
+#define IDB_BITMAP1 105
+#define IDD_WELCOME 106
+#define IDI_PROFILERED 106
+#define IDD_SELECTDB 107
+#define IDD_OPENERROR 108
+#define IDD_FILEACCESS 109
+#define IDD_CLEANING 110
+#define IDD_PROGRESS 111
+#define IDD_FINISHED 112
+#define IDI_BAD 113
+#define IDD_LOGIN 113
+#define IDC_WATERMARK 1000
+#define IDC_TITLE 1001
+#define IDC_HDRLOGO 1002
+#define IDC_DBLIST 1003
+#define IDC_FILE 1004
+#define IDC_OTHER 1005
+#define IDC_ERRORTEXT 1006
+#define IDC_INUSE 1007
+#define IDC_BACKUP 1008
+#define IDC_AGGRESSIVE 1009
+#define IDC_ERASEHISTORY 1010
+#define IDC_CHECKONLY 1010
+#define IDC_MARKREAD 1011
+#define IDC_PROGRESS 1011
+#define IDC_STATUS 1012
+#define IDC_STBACKUP 1013
+#define IDC_LAUNCHMIRANDA 1015
+#define IDC_DBFILE 1016
+#define IDC_BACKUPFILE 1017
+#define IDC_CHECK1 1018
+#define IDC_CONVERTUTF 1018
+#define IDC_DBNAME 1019
+#define IDC_USERPASS 1021
+#define IDC_LANG 1022
+#define IDC_LOGININFO 1023
+#define IDC_STATIC -1
+
+// Next default values for new objects
+//
+#ifdef APSTUDIO_INVOKED
+#ifndef APSTUDIO_READONLY_SYMBOLS
+#define _APS_NEXT_RESOURCE_VALUE 114
+#define _APS_NEXT_COMMAND_VALUE 40001
+#define _APS_NEXT_CONTROL_VALUE 1024
+#define _APS_NEXT_SYMED_VALUE 101
+#endif
+#endif
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/selectdb.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/selectdb.cpp
new file mode 100644
index 0000000000..9db3517816
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/selectdb.cpp
@@ -0,0 +1,303 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+void GetProfileDirectory(TCHAR* szMirandaDir, TCHAR* szPath, int cbPath)
+{
+ TCHAR szProfileDir[MAX_PATH], szExpandedProfileDir[MAX_PATH], szMirandaBootIni[MAX_PATH];
+
+ lstrcpy(szMirandaBootIni,szMirandaDir);
+ lstrcat(szMirandaBootIni,_T("\\mirandaboot.ini"));
+ GetPrivateProfileString(_T("Database"),_T("ProfileDir"),_T("./Profiles"),szProfileDir,SIZEOF(szProfileDir),szMirandaBootIni);
+ ExpandEnvironmentStrings(szProfileDir,szExpandedProfileDir,SIZEOF(szExpandedProfileDir));
+ _tchdir(szMirandaDir);
+ if (!_tfullpath(szPath,szExpandedProfileDir,cbPath))
+ lstrcpyn(szPath,szMirandaDir,cbPath);
+ if(szPath[lstrlen(szPath)-1]=='\\')
+ szPath[lstrlen(szPath)-1] = 0;
+}
+
+static int AddDatabaseToList(HWND hwndList, TCHAR* filename, TCHAR* dir)
+{
+ LV_ITEM lvi;
+ int iNewItem;
+ TCHAR szSize[20], *pName, *pDot, szName[MAX_PATH];
+ HANDLE hDbFile;
+ DBHeader dbhdr;
+ DWORD bytesRead;
+ DWORD totalSize,wasted=0;
+ int broken=0;
+
+ lvi.mask=LVIF_PARAM;
+ lvi.iSubItem=0;
+ for(lvi.iItem=ListView_GetItemCount(hwndList)-1;lvi.iItem>=0;lvi.iItem--) {
+ ListView_GetItem(hwndList,&lvi);
+ if ( !_tcsicmp(( TCHAR* )lvi.lParam,filename)) return lvi.iItem;
+ }
+ hDbFile=CreateFile(filename,GENERIC_READ,FILE_SHARE_READ|FILE_SHARE_WRITE,NULL,OPEN_EXISTING,0,NULL);
+ if ( hDbFile == INVALID_HANDLE_VALUE )
+ return -1;
+
+ ReadFile(hDbFile,&dbhdr,sizeof(dbhdr),&bytesRead,NULL);
+ totalSize=GetFileSize(hDbFile,NULL);
+
+ broken = (bytesRead<sizeof(dbhdr) || memcmp(dbhdr.signature,&dbSignature,sizeof(dbhdr.signature)));
+
+ if (!broken) {
+ wasted = dbhdr.slackSpace;
+ if (totalSize>dbhdr.ofsFileEnd)
+ wasted+=totalSize-dbhdr.ofsFileEnd;
+ }
+ CloseHandle(hDbFile);
+
+ pName = _tcsrchr(filename,'\\');
+ if ( pName == NULL )
+ pName = ( LPTSTR )filename;
+ else
+ pName++;
+ _tcscpy(szName,dir);
+ _tcscat(szName,pName);
+ pDot = _tcsrchr( szName, '.' );
+ if ( pDot != NULL && !_tcsicmp( pDot, _T(".dat")))
+ *pDot=0;
+
+ lvi.iItem = 0;
+ lvi.mask = LVIF_TEXT | LVIF_PARAM | LVIF_IMAGE;
+ lvi.iSubItem = 0;
+ lvi.lParam = ( LPARAM )_tcsdup(filename);
+ lvi.pszText = szName;
+ if (broken)
+ lvi.iImage = 3;
+ else if ( wasted < 1024*128 )
+ lvi.iImage = 0;
+ else if ( wasted < 1024*256 + (DWORD)(totalSize > 2*1024*1024 ) ? 256 * 1024 : 0 )
+ lvi.iImage=1;
+ else
+ lvi.iImage=2;
+
+ iNewItem = ListView_InsertItem(hwndList, &lvi );
+ _stprintf(szSize,_T("%.2lf MB"),totalSize/1048576.0);
+ ListView_SetItemText(hwndList,iNewItem,1,szSize);
+ if ( !broken ) {
+ _stprintf(szSize,_T("%.2lf MB"),wasted/1048576.0);
+ ListView_SetItemText(hwndList,iNewItem,2,szSize);
+ }
+ return iNewItem;
+}
+
+void FindAdd(HWND hdlg, TCHAR *szProfileDir, TCHAR *szPrefix)
+{
+ HANDLE hFind;
+ WIN32_FIND_DATA fd;
+ TCHAR szSearchPath[MAX_PATH],szFilename[MAX_PATH];
+
+ lstrcpy(szSearchPath,szProfileDir);
+ lstrcat(szSearchPath,_T("\\*.*"));
+
+ hFind=FindFirstFile(szSearchPath,&fd);
+ if (hFind!=INVALID_HANDLE_VALUE) {
+ do {
+ if ((fd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) && _tcscmp(fd.cFileName, _T(".")) && _tcscmp(fd.cFileName, _T(".."))) {
+ wsprintf(szFilename,_T("%s\\%s\\%s.dat"),szProfileDir,fd.cFileName,fd.cFileName);
+ if (_taccess(szFilename, 0) == 0)
+ AddDatabaseToList(GetDlgItem(hdlg,IDC_DBLIST),szFilename,szPrefix);
+ }
+ } while(FindNextFile(hFind,&fd));
+ FindClose(hFind);
+ }
+}
+
+TCHAR *addstring(TCHAR *str, TCHAR *add) {
+ _tcscpy(str,add);
+ return str + _tcslen(add) + 1;
+}
+
+INT_PTR CALLBACK SelectDbDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam)
+{
+ INT_PTR bReturn;
+ if ( DoMyControlProcessing( hdlg, message, wParam, lParam, &bReturn ))
+ return bReturn;
+
+ switch ( message ) {
+ case WM_INITDIALOG:
+ {
+ TCHAR szMirandaPath[MAX_PATH];
+ szMirandaPath[ 0 ] = 0;
+ { HIMAGELIST hIml;
+ hIml=ImageList_Create(GetSystemMetrics(SM_CXSMICON), GetSystemMetrics(SM_CYSMICON),
+ (IsWinVerXPPlus() ? ILC_COLOR32 : ILC_COLOR16) | ILC_MASK, 3, 3);
+ ImageList_AddIcon(hIml,LoadIcon(hInst,MAKEINTRESOURCE(IDI_PROFILEGREEN)));
+ ImageList_AddIcon(hIml,LoadIcon(hInst,MAKEINTRESOURCE(IDI_PROFILEYELLOW)));
+ ImageList_AddIcon(hIml,LoadIcon(hInst,MAKEINTRESOURCE(IDI_PROFILERED)));
+ ImageList_AddIcon(hIml,LoadIcon(hInst,MAKEINTRESOURCE(IDI_BAD)));
+ ListView_SetImageList(GetDlgItem(hdlg,IDC_DBLIST),hIml,LVSIL_SMALL);
+ }
+ ListView_SetExtendedListViewStyleEx(GetDlgItem(hdlg,IDC_DBLIST),LVS_EX_FULLROWSELECT,LVS_EX_FULLROWSELECT);
+ { LV_COLUMN lvc;
+ lvc.mask = LVCF_WIDTH | LVCF_FMT | LVCF_TEXT;
+ lvc.cx = 205;
+ lvc.fmt = LVCFMT_LEFT;
+ lvc.pszText = TranslateT("Database");
+ ListView_InsertColumn( GetDlgItem(hdlg,IDC_DBLIST), 0, &lvc );
+ lvc.cx = 68;
+ lvc.fmt = LVCFMT_RIGHT;
+ lvc.pszText = TranslateT("Total size");
+ ListView_InsertColumn(GetDlgItem(hdlg,IDC_DBLIST), 1, &lvc );
+ lvc.pszText = TranslateT("Wasted");
+ ListView_InsertColumn(GetDlgItem(hdlg,IDC_DBLIST), 2, &lvc );
+ }
+ {
+ TCHAR *str2;
+ GetModuleFileName(NULL,szMirandaPath,SIZEOF(szMirandaPath));
+ str2 = _tcsrchr(szMirandaPath,'\\');
+ if ( str2 != NULL )
+ *str2=0;
+ }
+ {
+ int i = 0;
+ HKEY hKey;
+ TCHAR szProfileDir[MAX_PATH];
+ DWORD cbData = SIZEOF(szMirandaPath);
+ TCHAR szMirandaProfiles[MAX_PATH];
+
+ _tcscpy(szMirandaProfiles, szMirandaPath);
+ _tcscat(szMirandaProfiles, _T("\\Profiles"));
+ GetProfileDirectory(szMirandaPath,szProfileDir,SIZEOF(szProfileDir));
+
+ // search in profile dir (using ini file)
+ if ( lstrcmpi(szProfileDir,szMirandaProfiles))
+ FindAdd(hdlg, szProfileDir, _T("[ini]\\"));
+
+ FindAdd(hdlg, szMirandaProfiles, _T("[prf]\\"));
+ // search in current dir (as DBTOOL)
+ FindAdd(hdlg, szMirandaPath, _T("[ . ]\\"));
+
+ // search in profile dir (using registry path + ini file)
+ if(RegOpenKeyEx(HKEY_LOCAL_MACHINE,_T("Software\\Microsoft\\Windows\\CurrentVersion\\App Paths\\miranda32.exe"),0,KEY_QUERY_VALUE,&hKey) == ERROR_SUCCESS) {
+ if(RegQueryValueEx(hKey,_T("Path"),NULL,NULL,(PBYTE)szMirandaPath,&cbData) == ERROR_SUCCESS) {
+ if ( lstrcmp(szProfileDir,szMirandaPath)) {
+ GetProfileDirectory(szMirandaPath,szProfileDir,SIZEOF(szProfileDir));
+ FindAdd(hdlg, szProfileDir, _T("[reg]\\"));
+ }
+ }
+ RegCloseKey(hKey);
+ }
+ // select
+ if ( opts.filename[0] )
+ i = AddDatabaseToList( GetDlgItem( hdlg, IDC_DBLIST ), opts.filename, _T(""));
+ if ( i == -1 )
+ i = 0;
+ ListView_SetItemState( GetDlgItem(hdlg,IDC_DBLIST), i, LVIS_SELECTED, LVIS_SELECTED );
+ }
+ if ( opts.hFile != NULL && opts.hFile != INVALID_HANDLE_VALUE ) {
+ CloseHandle( opts.hFile );
+ opts.hFile = NULL;
+ }
+ TranslateDialogDefault( hdlg );
+ return TRUE;
+ }
+
+ case WZN_PAGECHANGING:
+ GetDlgItemText( hdlg, IDC_FILE, opts.filename, SIZEOF(opts.filename));
+ break;
+
+ case WM_COMMAND:
+ switch(LOWORD(wParam)) {
+ case IDC_FILE:
+ if(HIWORD(wParam)==EN_CHANGE)
+ EnableWindow(GetDlgItem(GetParent(hdlg),IDOK),GetWindowTextLength(GetDlgItem(hdlg,IDC_FILE)));
+ break;
+ case IDC_OTHER:
+ { OPENFILENAME ofn={0};
+ TCHAR str[MAX_PATH];
+
+ // _T("Miranda Databases (*.dat)\0*.DAT\0All Files (*)\0*\0");
+ TCHAR *filter, *tmp, *tmp1, *tmp2;
+ tmp1 = TranslateT("Miranda Databases (*.dat)");
+ tmp2 = TranslateT("All Files");
+ filter = tmp = (TCHAR*)_alloca((_tcslen(tmp1)+_tcslen(tmp2)+11)*sizeof(TCHAR));
+ tmp = addstring(tmp, tmp1);
+ tmp = addstring(tmp, _T("*.DAT"));
+ tmp = addstring(tmp, tmp2);
+ tmp = addstring(tmp, _T("*"));
+ *tmp = 0;
+
+ GetDlgItemText( hdlg, IDC_FILE, str, SIZEOF( str ));
+ ofn.lStructSize = sizeof(ofn);
+ ofn.hwndOwner = hdlg;
+ ofn.hInstance = NULL;
+ ofn.lpstrFilter = filter;
+ ofn.lpstrDefExt = _T("dat");
+ ofn.lpstrFile = str;
+ ofn.Flags = OFN_FILEMUSTEXIST | OFN_HIDEREADONLY;
+ ofn.nMaxFile = SIZEOF(str);
+ ofn.nMaxFileTitle = MAX_PATH;
+ if ( GetOpenFileName( &ofn )) {
+ int i;
+ i = AddDatabaseToList( GetDlgItem(hdlg,IDC_DBLIST), str, _T(""));
+ if ( i == -1 )
+ i=0;
+ ListView_SetItemState( GetDlgItem(hdlg,IDC_DBLIST), i, LVIS_SELECTED, LVIS_SELECTED );
+ }
+ break;
+ }
+ case IDC_BACK:
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_WELCOME,(LPARAM)WelcomeDlgProc);
+ break;
+ case IDOK:
+ opts.hFile = CreateFile( opts.filename, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL );
+ if ( opts.hFile == INVALID_HANDLE_VALUE ) {
+ opts.hFile = NULL;
+ opts.error = GetLastError();
+ SendMessage( GetParent(hdlg), WZM_GOTOPAGE, IDD_OPENERROR, ( LPARAM )OpenErrorDlgProc );
+ }
+ else SendMessage( GetParent(hdlg), WZM_GOTOPAGE, IDD_FILEACCESS, (LPARAM)FileAccessDlgProc );
+ break;
+ }
+ break;
+ case WM_NOTIFY:
+ switch(((LPNMHDR)lParam)->idFrom) {
+ case IDC_DBLIST:
+ switch(((LPNMLISTVIEW)lParam)->hdr.code) {
+ case LVN_ITEMCHANGED:
+ { LV_ITEM lvi;
+ lvi.iItem=ListView_GetNextItem(GetDlgItem(hdlg,IDC_DBLIST),-1,LVNI_SELECTED);
+ if(lvi.iItem==-1) break;
+ lvi.mask=LVIF_PARAM;
+ ListView_GetItem(GetDlgItem(hdlg,IDC_DBLIST),&lvi);
+ SetDlgItemText(hdlg,IDC_FILE,(TCHAR*)lvi.lParam);
+ SendMessage(hdlg,WM_COMMAND,MAKEWPARAM(IDC_FILE,EN_CHANGE),(LPARAM)GetDlgItem(hdlg,IDC_FILE));
+ break;
+ }
+ }
+ break;
+ }
+ break;
+ case WM_DESTROY:
+ { LV_ITEM lvi;
+ lvi.mask=LVIF_PARAM;
+ for(lvi.iItem=ListView_GetItemCount(GetDlgItem(hdlg,IDC_DBLIST))-1;lvi.iItem>=0;lvi.iItem--) {
+ ListView_GetItem(GetDlgItem(hdlg,IDC_DBLIST),&lvi);
+ free((char*)lvi.lParam);
+ }
+ }
+ break;
+ }
+ return FALSE;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/settingschain.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/settingschain.cpp
new file mode 100644
index 0000000000..a6e7ecab3b
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/settingschain.cpp
@@ -0,0 +1,72 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+static DWORD ofsThisSettings,ofsDestPrevSettings;
+
+int WorkSettingsChain(DWORD ofsContact,DBContact *dbc,int firstTime)
+{
+ DBContactSettings *dbcsNew,dbcsOld;
+ DWORD ofsDestThis;
+ int ret;
+
+ if(firstTime) {
+ ofsDestPrevSettings=0;
+ ofsThisSettings=dbc->ofsFirstSettings;
+ dbc->ofsFirstSettings=0;
+ }
+ if(ofsThisSettings==0)
+ return ERROR_NO_MORE_ITEMS;
+ if (!SignatureValid(ofsThisSettings,DBCONTACTSETTINGS_SIGNATURE)) {
+ AddToStatus(STATUS_ERROR,TranslateT("Settings chain corrupted, further entries ignored"));
+ return ERROR_NO_MORE_ITEMS;
+ }
+ if(PeekSegment(ofsThisSettings,&dbcsOld,sizeof(dbcsOld))!=ERROR_SUCCESS)
+ return ERROR_NO_MORE_ITEMS;
+ if(dbcsOld.cbBlob>256*1024 || dbcsOld.cbBlob==0) {
+ AddToStatus(STATUS_ERROR,TranslateT("Infeasibly large settings blob: skipping"));
+ ofsThisSettings=dbcsOld.ofsNext;
+ return ERROR_SUCCESS;
+ }
+ dbcsNew=(DBContactSettings*)_alloca(offsetof(DBContactSettings,blob)+dbcsOld.cbBlob);
+ if ((ret=ReadSegment(ofsThisSettings,dbcsNew,offsetof(DBContactSettings,blob)+dbcsOld.cbBlob))!=ERROR_SUCCESS) {
+ if(ret!=ERROR_HANDLE_EOF) { //eof is OK because blank space at the end doesn't matter
+ return ERROR_NO_MORE_ITEMS;
+ }
+ }
+ if ((dbcsNew->ofsModuleName=ConvertModuleNameOfs(dbcsOld.ofsModuleName))==0) {
+ ofsThisSettings=dbcsOld.ofsNext;
+ return ERROR_SUCCESS;
+ }
+ if(dbcsNew->blob[0]==0) {
+ AddToStatus(STATUS_MESSAGE,TranslateT("Empty settings group at %08X: skipping"),ofsThisSettings);
+ ofsThisSettings=dbcsOld.ofsNext;
+ return ERROR_SUCCESS;
+ }
+ dbcsNew->ofsNext=0;
+ //TODO? validate all settings in blob/compact if necessary
+ if ((ofsDestThis=WriteSegment(WSOFS_END,dbcsNew,offsetof(DBContactSettings,blob)+dbcsNew->cbBlob))==WS_ERROR) {
+ return ERROR_HANDLE_DISK_FULL;
+ }
+ if(ofsDestPrevSettings) WriteSegment(ofsDestPrevSettings+offsetof(DBContactSettings,ofsNext),&ofsDestThis,sizeof(DWORD));
+ else dbc->ofsFirstSettings=ofsDestThis;
+ ofsDestPrevSettings=ofsDestThis;
+ ofsThisSettings=dbcsOld.ofsNext;
+ return ERROR_SUCCESS;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/user.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/user.cpp
new file mode 100644
index 0000000000..9922f1630f
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/user.cpp
@@ -0,0 +1,71 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+static DBContact user;
+static int phase;
+static DWORD ofsUser;
+
+int WorkSettingsChain(DWORD ofsContact,DBContact *dbc,int firstTime);
+int WorkEventChain(DWORD ofsContact,DBContact *dbc,int firstTime);
+
+int WorkUser(int firstTime)
+{
+ int first=0;
+
+ if(firstTime) {
+ AddToStatus(STATUS_MESSAGE,TranslateT("Processing user data"));
+ if (!SignatureValid(dbhdr.ofsUser,DBCONTACT_SIGNATURE)) {
+ AddToStatus(STATUS_ERROR,TranslateT("User corrupted, this could cause major problems"));
+ return ERROR_NO_MORE_ITEMS;
+ }
+ if(ReadSegment(dbhdr.ofsUser,&user,sizeof(DBContact))!=ERROR_SUCCESS)
+ return ERROR_NO_MORE_ITEMS;
+ if(user.ofsNext) {
+ AddToStatus(STATUS_WARNING,TranslateT("More than one user contact: keeping only first"));
+ user.ofsNext=0;
+ }
+ if ((ofsUser=WriteSegment(WSOFS_END,&user,sizeof(DBContact)))==WS_ERROR)
+ return ERROR_HANDLE_DISK_FULL;
+ dbhdr.ofsUser=ofsUser;
+ phase=0;
+ first=1;
+ }
+ switch(phase) {
+ int ret;
+
+ case 0:
+ ret=WorkSettingsChain(ofsUser,&user,first);
+ if(ret==ERROR_NO_MORE_ITEMS) {
+ phase++; first=1;
+ }
+ else if(ret) return ret;
+ else break;
+ case 1:
+ ret=WorkEventChain(ofsUser,&user,first);
+ if(ret==ERROR_NO_MORE_ITEMS) {
+ if(WriteSegment(ofsUser,&user,sizeof(DBContact))==WS_ERROR)
+ return ERROR_HANDLE_DISK_FULL;
+ return ERROR_NO_MORE_ITEMS;
+ }
+ else if(ret) return ret;
+ break;
+ }
+ return ERROR_SUCCESS;
+} \ No newline at end of file
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/utf.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/utf.cpp
new file mode 100644
index 0000000000..c56db053d8
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/utf.cpp
@@ -0,0 +1,69 @@
+/*
+
+Miranda IM: the free IM client for Microsoft* Windows*
+
+Copyright 2000-2009 Miranda ICQ/IM project,
+all portions of this codebase are copyrighted to the people
+listed in contributors.txt.
+
+ Copyright 2000 Alexandre Julliard of Wine project
+ (UTF-8 conversion routines)
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+
+#include "dbtool.h"
+
+bool is_utf8_string(const char* str)
+{
+ int expect_bytes = 0, utf_found = 0;
+
+ if (!str) return 0;
+
+ while (*str) {
+ if ((*str & 0x80) == 0) {
+ /* Looks like an ASCII character */
+ if (expect_bytes)
+ /* byte of UTF-8 character expected */
+ return 0;
+ }
+ else {
+ /* Looks like byte of an UTF-8 character */
+ if (expect_bytes) {
+ /* expect_bytes already set: first byte of UTF-8 char already seen */
+ if ((*str & 0xC0) != 0x80) {
+ /* again first byte ?!?! */
+ return 0;
+ }
+ }
+ else {
+ /* First byte of the UTF-8 character */
+ /* count initial one bits and set expect_bytes to 1 less */
+ char ch = *str;
+ while (ch & 0x80) {
+ expect_bytes++;
+ ch = (ch & 0x7f) << 1;
+ }
+ }
+ /* OK, next byte of UTF-8 character */
+ /* Decrement number of expected bytes */
+ if ( --expect_bytes == 0 )
+ utf_found = 1;
+ }
+ str++;
+ }
+
+ return (utf_found && expect_bytes == 0);
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/welcome.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/welcome.cpp
new file mode 100644
index 0000000000..291281e70a
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/welcome.cpp
@@ -0,0 +1,61 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+INT_PTR CALLBACK WelcomeDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam)
+{
+ static HENHMETAFILE hEmfWatermark;
+ static HFONT hTitleFont;
+ INT_PTR bReturn;
+
+ if(DoMyControlProcessing(hdlg,message,wParam,lParam,&bReturn)) return bReturn;
+ switch(message) {
+ case WM_INITDIALOG:
+
+ {
+ NONCLIENTMETRICS ncm = {0};
+ ncm.cbSize = sizeof(ncm);
+ SystemParametersInfo(SPI_GETNONCLIENTMETRICS, 0, &ncm, 0);
+ LOGFONT TitleLogFont = ncm.lfMessageFont;
+ TitleLogFont.lfWeight = FW_BOLD;
+ lstrcpy(TitleLogFont.lfFaceName, TEXT("Verdana Bold"));
+
+ HDC hdc = GetDC(NULL);
+ INT FontSize = 12;
+ TitleLogFont.lfHeight = 0 - GetDeviceCaps(hdc, LOGPIXELSY) * FontSize / 72;
+ hTitleFont = CreateFontIndirect(&TitleLogFont);
+ ReleaseDC(NULL, hdc);
+ }
+ SendDlgItemMessage(hdlg,IDC_TITLE,WM_SETFONT,(WPARAM)hTitleFont,0);
+ EnableWindow(GetDlgItem(GetParent(hdlg),IDC_BACK),FALSE);
+ return FALSE;
+ case WM_COMMAND:
+ switch(LOWORD(wParam)) {
+ case IDOK:
+ SendMessage(GetParent(hdlg),WZM_GOTOPAGE,IDD_SELECTDB,(LPARAM)SelectDbDlgProc);
+ break;
+ }
+ break;
+ case WM_DESTROY:
+ DeleteEnhMetaFile(hEmfWatermark);
+ DeleteObject(hTitleFont);
+ break;
+ }
+ return FALSE;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/wizard.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/wizard.cpp
new file mode 100644
index 0000000000..8f3a9d5ca8
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/wizard.cpp
@@ -0,0 +1,137 @@
+/*
+Miranda Database Tool
+Copyright 2000-2011 Miranda ICQ/IM project,
+all portions of this codebase are copyrighted to the people
+listed in contributors.txt.
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+static HFONT hBoldFont=NULL;
+static HENHMETAFILE hEmfHeaderLogo=NULL;
+
+static BOOL CALLBACK MyControlsEnumChildren(HWND hwnd,LPARAM lParam)
+{
+ DWORD style=GetWindowLongPtr(hwnd,GWL_STYLE);
+ DWORD exstyle=GetWindowLongPtr(hwnd,GWL_EXSTYLE);
+ char szClass[64];
+ int makeBold=0;
+
+ GetClassNameA(hwnd,szClass,sizeof(szClass));
+ if (!strcmp(szClass,"Static")) {
+ if (((style&SS_TYPEMASK)==SS_LEFT || (style&SS_TYPEMASK)==SS_CENTER || (style&SS_TYPEMASK)==SS_RIGHT) && exstyle&WS_EX_CLIENTEDGE)
+ makeBold=1;
+ }
+ else if (!strcmp(szClass,"Button")) {
+ if(exstyle&WS_EX_CLIENTEDGE)
+ makeBold=1;
+ }
+ if(makeBold) {
+ if(hBoldFont==NULL) {
+ LOGFONT lf;
+ hBoldFont=(HFONT)SendMessage(hwnd,WM_GETFONT,0,0);
+ GetObject(hBoldFont,sizeof(lf),&lf);
+ lf.lfWeight=FW_BOLD;
+ hBoldFont=CreateFontIndirect(&lf);
+ }
+ SendMessage(hwnd,WM_SETFONT,(WPARAM)hBoldFont,0);
+ SetWindowLongPtr(hwnd,GWL_EXSTYLE,exstyle&~WS_EX_CLIENTEDGE);
+ SetWindowPos(hwnd,0,0,0,0,0,SWP_NOZORDER|SWP_NOMOVE|SWP_NOSIZE|SWP_FRAMECHANGED);
+ }
+ return TRUE;
+}
+
+int DoMyControlProcessing(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam,INT_PTR *bReturn)
+{
+ switch(message) {
+ case WM_INITDIALOG:
+ EnumChildWindows(hdlg,MyControlsEnumChildren,0);
+
+ break;
+ case WM_CTLCOLORSTATIC:
+ if (GetWindowLongPtr((HWND)lParam, GWL_ID) == IDC_HDRLOGO){
+ SetBkColor((HDC)wParam,RGB(255,255,255));
+ *bReturn=(BOOL)GetStockObject(WHITE_BRUSH);
+ return TRUE;
+ }
+ if ((GetWindowLongPtr((HWND)lParam,GWL_STYLE)&0xFFFF)==0) {
+ TCHAR szText[256];
+ GetWindowText((HWND)lParam,szText,SIZEOF(szText));
+ if (!_tcscmp(szText,_T("whiterect"))) {
+ SetTextColor((HDC)wParam,RGB(255,255,255));
+ SetBkColor((HDC)wParam,RGB(255,255,255));
+ SetBkMode((HDC)wParam,OPAQUE);
+ *bReturn=(INT_PTR)GetStockObject(WHITE_BRUSH);
+ return TRUE;
+ }
+ else {
+ SetBkMode((HDC)wParam,TRANSPARENT);
+ *bReturn=(INT_PTR)GetStockObject(NULL_BRUSH);
+ return TRUE;
+ }
+ }
+ break;
+ }
+ return FALSE;
+}
+
+INT_PTR CALLBACK WizardDlgProc(HWND hdlg,UINT message,WPARAM wParam,LPARAM lParam)
+{
+ static HWND hdlgPage;
+
+ switch(message) {
+ case WM_INITDIALOG:
+ SendMessage(hdlg,WM_SETICON,ICON_SMALL,(LPARAM)LoadIcon(hInst,MAKEINTRESOURCE(IDI_DBTOOL)));
+ hdlgPage=NULL;
+ SendMessage(hdlg,WZM_GOTOPAGE,IDD_WELCOME,(LPARAM)WelcomeDlgProc);
+ TranslateDialogDefault(hdlg);
+ return TRUE;
+ case WZM_GOTOPAGE:
+ if(hdlgPage!=NULL) DestroyWindow(hdlgPage);
+ EnableWindow(GetDlgItem(hdlg,IDC_BACK),TRUE);
+ EnableWindow(GetDlgItem(hdlg,IDOK),TRUE);
+ EnableWindow(GetDlgItem(hdlg,IDCANCEL),TRUE);
+ SetDlgItemText(hdlg,IDCANCEL,TranslateT("Cancel"));
+ hdlgPage=CreateDialog(hInst,MAKEINTRESOURCE(wParam),hdlg,(DLGPROC)lParam);
+ TranslateDialogDefault(hdlgPage);
+ SetWindowPos(hdlgPage,0,0,0,0,0,SWP_NOZORDER|SWP_NOSIZE);
+ ShowWindow(hdlgPage,SW_SHOW);
+ break;
+ case WM_COMMAND:
+ switch(LOWORD(wParam)) {
+ case IDC_BACK:
+ case IDOK:
+ SendMessage(hdlgPage,WZN_PAGECHANGING,wParam,0);
+ SendMessage(hdlgPage,message,wParam,lParam);
+ break;
+ case IDCANCEL:
+ if(SendMessage(hdlgPage,WZN_CANCELCLICKED,0,0)) break;
+ EndDialog(hdlg,0);
+ break;
+ }
+ break;
+ case WM_DESTROY:
+ if (opts.hFile)
+ CloseHandle(opts.hFile);
+ if (opts.hOutFile)
+ CloseHandle(opts.hOutFile);
+ DestroyWindow(hdlgPage);
+ if(hBoldFont!=NULL) DeleteObject(hBoldFont);
+ if(hEmfHeaderLogo!=NULL) DeleteEnhMetaFile(hEmfHeaderLogo);
+ break;
+ }
+ return FALSE;
+}
diff --git a/plugins/Dbx_mmap_SA/Dbtool/src/worker.cpp b/plugins/Dbx_mmap_SA/Dbtool/src/worker.cpp
new file mode 100644
index 0000000000..dd73c83947
--- /dev/null
+++ b/plugins/Dbx_mmap_SA/Dbtool/src/worker.cpp
@@ -0,0 +1,76 @@
+/*
+Miranda Database Tool
+Copyright (C) 2001-2005 Richard Hughes
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation; either version 2
+of the License, or (at your option) any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+#include "dbtool.h"
+
+void ProcessingDone(void);
+int WorkInitialChecks(int firstTime);
+int WorkModuleChain(int firstTime);
+int WorkUser(int firstTime);
+int WorkContactChain(int firstTime);
+int WorkAggressive(int firstTime);
+int WorkFinalTasks(int firstTime);
+
+extern HANDLE hEventRun,hEventAbort;
+extern int errorCount;
+time_t ts;
+DBHeader dbhdr;
+DWORD spaceProcessed,sourceFileSize;
+DWORD spaceUsed, sp;
+
+static int (*Workers[6])(int)=
+ {WorkInitialChecks,WorkModuleChain,WorkUser,WorkContactChain,WorkAggressive,WorkFinalTasks};
+
+void __cdecl WorkerThread(void *unused)
+{
+ int task,ret,firstTime;
+ ts=time(NULL);
+
+ AddToStatus(STATUS_MESSAGE,TranslateT("Database worker thread activated"));
+ SetFilePointer(opts.hFile,0,NULL,FILE_BEGIN);
+ spaceUsed=1; spaceProcessed=0; sp=0;
+ firstTime=0;
+
+ for(task=0;;) {
+ if (spaceProcessed/(spaceUsed/1000+1) > sp) {
+ sp = spaceProcessed/(spaceUsed/1000+1);
+ SetProgressBar(sp);
+ }
+ WaitForSingleObject(hEventRun,INFINITE);
+ if(WaitForSingleObject(hEventAbort,0)==WAIT_OBJECT_0) {
+ AddToStatus(STATUS_FATAL,TranslateT("Processing aborted by user"));
+ break;
+ }
+ ret=Workers[task](firstTime);
+ firstTime=0;
+ if(ret==ERROR_NO_MORE_ITEMS) {
+ if (++task==sizeof(Workers)/sizeof(Workers[0])) {
+ AddToStatus(STATUS_MESSAGE,TranslateT("Elapsed time: %d sec"), time(NULL)-ts);
+ if(errorCount)
+ AddToStatus(STATUS_SUCCESS,TranslateT("All tasks completed but with errors (%d)"),errorCount);
+ else
+ AddToStatus(STATUS_SUCCESS,TranslateT("All tasks completed successfully"));
+ break;
+ }
+ firstTime=1;
+ }
+ else if(ret!=ERROR_SUCCESS)
+ break;
+ }
+ ProcessingDone();
+}