Restoring your win32 Hugs98 installations
Sigbjorn Finne
sof@galconn.com
Tue, 7 Aug 2001 10:14:29 -0700
This is a multi-part message in MIME format.
------=_NextPart_000_148F_01C11F29.BE7148E0
Content-Type: text/plain;
charset="iso-8859-1"
Content-Transfer-Encoding: 7bit
Hi,
attached is a little utility I cobbled together to help out someone
that had gotten their Hugs98 install on Win32 in a bad state,
and was having a hard time reverting it back into working order.
(In particular, the search path was ill-formed, causing Hugs to exit
before it could load the Prelude).
The utility lets you wipe out the user-specific Hugs98 Registry :settings,
which in effect reverts the Hugs98 settings back to the default ones.
A stop-gap solution that is hopefully of use to others.
hth
--sigbjorn
btw, if you don't have access to a C compiler to compile up the sources,
let me know & I'll mail you a binary.
------=_NextPart_000_148F_01C11F29.BE7148E0
Content-Type: application/octet-stream;
name="restore.c"
Content-Transfer-Encoding: quoted-printable
Content-Disposition: attachment;
filename="restore.c"
/*
* Hugs98 Registry reset app - get rid off the
* user-specific Hugs98 options settings from the Registry.
*
* The reason why this util is needed is that Hugs doesn't=20
* perform any validity checks when persisting the :set'able
* options to the Registry. As a result, the user might end
* up inadvertently zapping the search path, causing Hugs=20
* to be unable to locate the Prelude the next time it starts
* up. Which is undesirable, since it'll then exit before
* the useris able to get to get to the interpreter prompt
* & repair the damage done via another :set !!
*
* The only way to exit from this unhappy state of affairs
* is to either be Registry-savy and do the mods 'regedit',
* or reinstall. So, clearly some room for improvement in
* usability here.
*
* This util is a stop-gap measure which lets the user rid
* him/herself of the *user-specific* option settings that
* Hugs will have saved in the Registry. By deleting the
* user-specific settings, the next time Hugs starts up,
* it'll default back to using the machine-wide default
* options. i.e., the user may end up having to go in=20
* and tweak these settings again, but at least Hugs has
* been brought back into a useable state.
*
* To compile this utility, you'll need a Win32 C compiler;
* here are the options to use with the three compilers I've
* tested it with:
*
* MSVC: cl /o restorer restore.c advapi32.lib
* gcc: gcc -mno-cygwin -o restorer restore.c
* lcc-win32: lc restore.c -o main.exe
*=20
* 7/01 -- sof
*/
#include <windows.h>
#include <malloc.h>
#include <stdio.h>
#define MAX(x,y) ( x > y ? x : y)
#define HUGS_HCU_ROOT "Software\\Haskell\\Hugs"
/* This util inspects the Registry looking for
existing Hugs installations in the current
user hive.
Store each one of the installs found in a table,
so that we can later on go in and selectively
reset each one of them.
*/
typedef struct {
TCHAR* keyName;
HKEY hKey;
} HugsInstall;
typedef struct {
unsigned int size;
unsigned int idx;
HugsInstall* tab;
} InstallTable;
/*
* Function: initInstallTable ( unsigned int sz )
*
* Purpose: Allocate a table (of 'sz' elements) that will
* hold information about available Hugs installation.
*
* Returns: empty Hugs InstallTable of size 'sz' elements.
*/
InstallTable*
initInstallTable(unsigned int size)
{
HugsInstall* tab;
InstallTable* table;
=20
table =3D (InstallTable*)malloc(sizeof(InstallTable));
=20
if (table =3D=3D NULL) {
return table;
}
tab =3D (HugsInstall*)malloc(size * sizeof(HugsInstall));
=20
if (tab =3D=3D NULL) {
free(table);
return NULL;
}
=20
table->size =3D size;
table->idx =3D 0;
table->tab =3D tab;
=20
return table;
}
void
DestroyTable(InstallTable* table)
{
int i;
if (!table)=20
return;
for ( i=3D0; i < (int)table->idx ; i++ ) {
free(table->tab[i].keyName);
RegCloseKey(table->tab[i].hKey);
}
=20
free(table->tab);
free(table);
} =20
/*
* Function: expandTable(InstallTable*)
*
* Purpose: expand the installation table by 50%.
* Helper routine used by addInstall().
*
* Returns: boolean indicating whether the expansion
* was successful or not.
*/
BOOL
expandTable(InstallTable* table)
{
HugsInstall* tab;
unsigned int newSz;
if (!table) {
return FALSE;
}
=20
newSz =3D table->size + MAX(1, (table -> size / 2));
tab =3D (HugsInstall*)realloc(table->tab, sizeof(HugsInstall) * =
newSz);
if (tab =3D=3D NULL) {
return FALSE;
}
table->size =3D newSz;
table->tab =3D tab;
return TRUE;
}
BOOL
addHugsInstall(InstallTable* table, char* dirName, HKEY hKey)
{
char* tmp;
if (!table)
return FALSE;
/* idx points to the next available slot */
if ( table->idx =3D=3D table->size ) {
if ( !expandTable(table) ) {
return FALSE;
}
}
=20
tmp =3D (char*)malloc(sizeof(char) * (strlen(dirName) + 1));
=20
if (!tmp)=20
return FALSE;
=20
strcpy(tmp, dirName);
table->tab[table->idx].keyName =3D tmp;
table->tab[table->idx].hKey =3D hKey;
=20
table->idx +=3D 1;
=20
return TRUE;
}
#define HandleRegError(msg, rc) printf("Error calling %s - return code =
=3D 0x%x\n", msg, rc)
void
ResetOption(InstallTable* table, int idx)
{
LONG lRes;
=20
lRes =3D RegDeleteValue(table->tab[idx].hKey, "Options");
printf("Finished resetting %s\n\n", table->tab[idx].keyName);
}
void
HandleDeletion(InstallTable* table)
{
int i, idx;
int ch;
int val;
char line[10];
BOOL* delVector;
int* idxVector;
=20
printf("Hugs98 Repair Tool\n\n");
printf("By selecting one of the installations below, you'll reset\n");
printf("your user-specific options that Hugs98 has saved away\n");
printf("in the Windows Registry (Resetting just gets you back\n");
printf("to using the default Hugs options).\n\n");
=20
if (!table || table->idx =3D=3D 0) {
printf("\n\n No local Hugs installations found. Exiting\n");
return;
}
=20
idxVector=3D(int*)malloc(sizeof(int) * (table->idx));
delVector=3D(int*)malloc(sizeof(BOOL) * (table->idx));
for (i=3D0;i < (int)table->idx; i++) {
delVector[i]=3DFALSE;
idxVector[i]=3D-1;
}
do {
idx =3D 0;
printf("Local Hugs98 installations: \n");
for (i=3D0; i < (int)table->idx ; i++ ) {
if (delVector[i]) {
continue;
}
idxVector[idx] =3D i;
printf(" %i - %s\n", idx+1, table->tab[i].keyName);
idx++;
}
=20
printf("\nInstallation to reset (0 =3D=3D reset all; ENTER to exit) =
? ");
fflush(stdout);
if (fgets(line,10,stdin) !=3D NULL) {
if (line[0] =3D=3D '\n') {
free(delVector);
free(idxVector);
return;
}
val =3D atoi(line);
if (val !=3D 0) {
if ( val > idx ) {
printf("Selection, %u, out of range\n", val);
continue;
}
ResetOption(table, idxVector[val-1]);
delVector[idxVector[val-1]]=3DTRUE;
} else {
for (i=3D0; i < table->idx ; i++) {
if (!delVector[i]) {
ResetOption(table, i);
}
}
free(delVector);
free(idxVector);
break;
}
}
} while (1);
}
int
main()
{
HKEY hHugsKey;
HKEY hSubKey;
LONG lRes;
TCHAR nameBuf [ 511 ];
TCHAR nameBuf2 [ 2048 ];
DWORD dwSize, dwIndex;
InstallTable* table;
FILETIME ft;
lRes =3D=20
RegOpenKeyEx ( HKEY_CURRENT_USER,
HUGS_HCU_ROOT,
0,
KEY_ALL_ACCESS,
&hHugsKey);
=20
if ( lRes !=3D ERROR_SUCCESS ) {
HandleRegError("RegOpenKey", lRes);
return 1;
}
=20
dwIndex =3D -1;
=20
table =3D initInstallTable(2);
do {
dwSize =3D sizeof ( nameBuf );
dwIndex++;
lRes =3D=20
RegEnumKeyEx ( hHugsKey,
dwIndex,
nameBuf,
&dwSize,
NULL,
NULL,
NULL,
&ft);
if ( lRes =3D=3D ERROR_SUCCESS ||
lRes =3D=3D ERROR_MORE_DATA ||=20
lRes =3D=3D ERROR_NO_MORE_ITEMS ) {
strcpy(nameBuf2, HUGS_HCU_ROOT);
strcat(nameBuf2, "\\");
strcat(nameBuf2, nameBuf);
RegOpenKeyEx ( HKEY_CURRENT_USER,
nameBuf2,
0,
KEY_ALL_ACCESS,
&hSubKey);
if ( !addHugsInstall(table, nameBuf, hSubKey) ) {
break;
}
}
} while ( lRes =3D=3D ERROR_SUCCESS ||=20
lRes =3D=3D ERROR_MORE_DATA );=20
=20
if ( lRes !=3D ERROR_MORE_DATA && lRes !=3D ERROR_NO_MORE_ITEMS ) {
HandleRegError("RegEnumKeyEx", lRes);
return 0;
}
=20
HandleDeletion(table);
=20
DestroyTable(table);
RegCloseKey(hHugsKey);
=20
return 0;
}
------=_NextPart_000_148F_01C11F29.BE7148E0--