2003-01-18 12:03:28 +00:00
|
|
|
/* $Id: macstore.c,v 1.9 2003/01/18 12:03:28 ben Exp $ */
|
2002-11-19 02:13:46 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* macstore.c: Macintosh-specific impementation of the interface
|
|
|
|
* defined in storage.h
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <MacTypes.h>
|
|
|
|
#include <Folders.h>
|
|
|
|
#include <Memory.h>
|
|
|
|
#include <Resources.h>
|
|
|
|
#include <TextUtils.h>
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "putty.h"
|
|
|
|
#include "storage.h"
|
2002-12-30 18:21:17 +00:00
|
|
|
#include "mac.h"
|
2002-11-19 02:13:46 +00:00
|
|
|
|
|
|
|
#define PUTTY_CREATOR FOUR_CHAR_CODE('pTTY')
|
|
|
|
#define SESS_TYPE FOUR_CHAR_CODE('Sess')
|
2003-01-08 22:46:12 +00:00
|
|
|
#define SEED_TYPE FOUR_CHAR_CODE('Seed')
|
2002-11-19 02:13:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
OSErr FSpGetDirID(FSSpec *f, long *idp, Boolean makeit);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We store each session as a file in the "PuTTY" sub-directory of the
|
|
|
|
* preferences folder. Each (key,value) pair is stored as a resource.
|
|
|
|
*/
|
|
|
|
|
2003-01-08 22:46:12 +00:00
|
|
|
OSErr get_putty_dir(Boolean makeit, short *pVRefNum, long *pDirID)
|
|
|
|
{
|
2002-11-19 02:13:46 +00:00
|
|
|
OSErr error = noErr;
|
|
|
|
short prefVRefNum;
|
2003-01-08 22:46:12 +00:00
|
|
|
FSSpec puttydir;
|
|
|
|
long prefDirID, puttyDirID;
|
2002-11-19 02:13:46 +00:00
|
|
|
|
|
|
|
error = FindFolder(kOnSystemDisk, kPreferencesFolderType, makeit,
|
|
|
|
&prefVRefNum, &prefDirID);
|
|
|
|
if (error != noErr) goto out;
|
|
|
|
|
|
|
|
error = FSMakeFSSpec(prefVRefNum, prefDirID, "\pPuTTY", &puttydir);
|
|
|
|
if (error != noErr && error != fnfErr) goto out;
|
|
|
|
error = FSpGetDirID(&puttydir, &puttyDirID, makeit);
|
|
|
|
if (error != noErr) goto out;
|
|
|
|
|
2003-01-08 22:46:12 +00:00
|
|
|
*pVRefNum = prefVRefNum;
|
|
|
|
*pDirID = puttyDirID;
|
|
|
|
|
|
|
|
out:
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
OSErr get_session_dir(Boolean makeit, short *pVRefNum, long *pDirID) {
|
|
|
|
OSErr error = noErr;
|
|
|
|
short puttyVRefNum;
|
|
|
|
FSSpec sessdir;
|
|
|
|
long puttyDirID, sessDirID;
|
|
|
|
|
|
|
|
error = get_putty_dir(makeit, &puttyVRefNum, &puttyDirID);
|
|
|
|
if (error != noErr) goto out;
|
|
|
|
error = FSMakeFSSpec(puttyVRefNum, puttyDirID, "\pSaved Sessions",
|
2002-11-19 02:13:46 +00:00
|
|
|
&sessdir);
|
|
|
|
if (error != noErr && error != fnfErr) goto out;
|
|
|
|
error = FSpGetDirID(&sessdir, &sessDirID, makeit);
|
|
|
|
if (error != noErr) goto out;
|
|
|
|
|
2003-01-08 22:46:12 +00:00
|
|
|
*pVRefNum = puttyVRefNum;
|
2002-11-19 02:13:46 +00:00
|
|
|
*pDirID = sessDirID;
|
|
|
|
|
|
|
|
out:
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
OSErr FSpGetDirID(FSSpec *f, long *idp, Boolean makeit) {
|
|
|
|
CInfoPBRec pb;
|
|
|
|
OSErr error = noErr;
|
|
|
|
|
|
|
|
pb.dirInfo.ioNamePtr = f->name;
|
|
|
|
pb.dirInfo.ioVRefNum = f->vRefNum;
|
|
|
|
pb.dirInfo.ioDrDirID = f->parID;
|
|
|
|
pb.dirInfo.ioFDirIndex = 0;
|
|
|
|
error = PBGetCatInfoSync(&pb);
|
|
|
|
if (error == fnfErr && makeit)
|
|
|
|
return FSpDirCreate(f, smSystemScript, idp);
|
|
|
|
if (error != noErr) goto out;
|
|
|
|
if ((pb.dirInfo.ioFlAttrib & ioDirMask) == 0) {
|
|
|
|
error = dirNFErr;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
*idp = pb.dirInfo.ioDrDirID;
|
|
|
|
|
|
|
|
out:
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct write_settings {
|
|
|
|
int fd;
|
|
|
|
FSSpec tmpfile;
|
|
|
|
FSSpec dstfile;
|
|
|
|
};
|
|
|
|
|
2003-01-14 19:09:24 +00:00
|
|
|
void *open_settings_w(char const *sessionname) {
|
2002-11-19 02:13:46 +00:00
|
|
|
short sessVRefNum, tmpVRefNum;
|
|
|
|
long sessDirID, tmpDirID;
|
|
|
|
OSErr error;
|
|
|
|
Str255 psessionname;
|
|
|
|
struct write_settings *ws;
|
|
|
|
|
|
|
|
ws = safemalloc(sizeof *ws);
|
|
|
|
error = get_session_dir(kCreateFolder, &sessVRefNum, &sessDirID);
|
|
|
|
if (error != noErr) goto out;
|
|
|
|
|
|
|
|
c2pstrcpy(psessionname, sessionname);
|
|
|
|
error = FSMakeFSSpec(sessVRefNum, sessDirID, psessionname, &ws->dstfile);
|
|
|
|
if (error != noErr && error != fnfErr) goto out;
|
|
|
|
if (error == fnfErr) {
|
|
|
|
FSpCreateResFile(&ws->dstfile, PUTTY_CREATOR, SESS_TYPE,
|
|
|
|
smSystemScript);
|
|
|
|
if ((error = ResError()) != noErr) goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Create a temporary file to save to first. */
|
|
|
|
error = FindFolder(sessVRefNum, kTemporaryFolderType, kCreateFolder,
|
|
|
|
&tmpVRefNum, &tmpDirID);
|
|
|
|
if (error != noErr) goto out;
|
|
|
|
error = FSMakeFSSpec(tmpVRefNum, tmpDirID, psessionname, &ws->tmpfile);
|
|
|
|
if (error != noErr && error != fnfErr) goto out;
|
|
|
|
if (error == noErr) {
|
|
|
|
error = FSpDelete(&ws->tmpfile);
|
|
|
|
if (error != noErr) goto out;
|
|
|
|
}
|
|
|
|
FSpCreateResFile(&ws->tmpfile, PUTTY_CREATOR, SESS_TYPE, smSystemScript);
|
|
|
|
if ((error = ResError()) != noErr) goto out;
|
|
|
|
|
|
|
|
ws->fd = FSpOpenResFile(&ws->tmpfile, fsWrPerm);
|
|
|
|
if (ws->fd == -1) {error = ResError(); goto out;}
|
|
|
|
|
|
|
|
return ws;
|
|
|
|
|
|
|
|
out:
|
|
|
|
safefree(ws);
|
|
|
|
fatalbox("Failed to open session for write (%d)", error);
|
|
|
|
}
|
|
|
|
|
2003-01-14 19:09:24 +00:00
|
|
|
void write_setting_s(void *handle, char const *key, char const *value) {
|
2002-11-19 02:13:46 +00:00
|
|
|
int fd = *(int *)handle;
|
|
|
|
Handle h;
|
|
|
|
int id;
|
2002-12-28 22:44:27 +00:00
|
|
|
OSErr error;
|
2002-11-19 02:13:46 +00:00
|
|
|
|
|
|
|
UseResFile(fd);
|
|
|
|
if (ResError() != noErr)
|
|
|
|
fatalbox("Failed to open saved session (%d)", ResError());
|
|
|
|
|
|
|
|
error = PtrToHand(value, &h, strlen(value));
|
|
|
|
if (error != noErr)
|
|
|
|
fatalbox("Failed to allocate memory");
|
|
|
|
/* Put the data in a resource. */
|
|
|
|
id = Unique1ID(FOUR_CHAR_CODE('TEXT'));
|
|
|
|
if (ResError() != noErr)
|
|
|
|
fatalbox("Failed to get ID for resource %s (%d)", key, ResError());
|
|
|
|
addresource(h, FOUR_CHAR_CODE('TEXT'), id, key);
|
|
|
|
if (ResError() != noErr)
|
|
|
|
fatalbox("Failed to add resource %s (%d)", key, ResError());
|
|
|
|
}
|
|
|
|
|
2003-01-14 19:09:24 +00:00
|
|
|
void write_setting_i(void *handle, char const *key, int value) {
|
2002-11-19 02:13:46 +00:00
|
|
|
int fd = *(int *)handle;
|
|
|
|
Handle h;
|
|
|
|
int id;
|
|
|
|
OSErr error;
|
|
|
|
|
|
|
|
UseResFile(fd);
|
|
|
|
if (ResError() != noErr)
|
|
|
|
fatalbox("Failed to open saved session (%d)", ResError());
|
|
|
|
|
|
|
|
/* XXX assume all systems have the same "int" format */
|
|
|
|
error = PtrToHand(&value, &h, sizeof(int));
|
|
|
|
if (error != noErr)
|
|
|
|
fatalbox("Failed to allocate memory (%d)", error);
|
|
|
|
|
|
|
|
/* Put the data in a resource. */
|
|
|
|
id = Unique1ID(FOUR_CHAR_CODE('Int '));
|
|
|
|
if (ResError() != noErr)
|
|
|
|
fatalbox("Failed to get ID for resource %s (%d)", key, ResError());
|
|
|
|
addresource(h, FOUR_CHAR_CODE('Int '), id, key);
|
|
|
|
if (ResError() != noErr)
|
|
|
|
fatalbox("Failed to add resource %s (%d)", key, ResError());
|
|
|
|
}
|
|
|
|
|
|
|
|
void close_settings_w(void *handle) {
|
|
|
|
struct write_settings *ws = handle;
|
|
|
|
OSErr error;
|
|
|
|
|
|
|
|
CloseResFile(ws->fd);
|
|
|
|
if ((error = ResError()) != noErr)
|
|
|
|
goto out;
|
|
|
|
error = FSpExchangeFiles(&ws->tmpfile, &ws->dstfile);
|
|
|
|
if (error != noErr) goto out;
|
|
|
|
error = FSpDelete(&ws->tmpfile);
|
|
|
|
if (error != noErr) goto out;
|
|
|
|
return;
|
|
|
|
|
|
|
|
out:
|
|
|
|
fatalbox("Close of saved session failed (%d)", error);
|
|
|
|
safefree(handle);
|
|
|
|
}
|
|
|
|
|
2003-01-14 19:09:24 +00:00
|
|
|
void *open_settings_r(char const *sessionname)
|
2002-12-30 18:21:17 +00:00
|
|
|
{
|
2002-11-19 02:13:46 +00:00
|
|
|
short sessVRefNum;
|
|
|
|
long sessDirID;
|
|
|
|
FSSpec sessfile;
|
|
|
|
OSErr error;
|
|
|
|
Str255 psessionname;
|
|
|
|
|
|
|
|
error = get_session_dir(kDontCreateFolder, &sessVRefNum, &sessDirID);
|
|
|
|
|
|
|
|
c2pstrcpy(psessionname, sessionname);
|
|
|
|
error = FSMakeFSSpec(sessVRefNum, sessDirID, psessionname, &sessfile);
|
|
|
|
if (error != noErr) goto out;
|
2002-12-30 18:21:17 +00:00
|
|
|
return open_settings_r_fsp(&sessfile);
|
|
|
|
|
|
|
|
out:
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *open_settings_r_fsp(FSSpec *sessfile)
|
|
|
|
{
|
|
|
|
OSErr error;
|
|
|
|
int fd;
|
|
|
|
int *handle;
|
|
|
|
|
|
|
|
fd = FSpOpenResFile(sessfile, fsRdPerm);
|
2002-11-19 02:13:46 +00:00
|
|
|
if (fd == 0) {error = ResError(); goto out;}
|
|
|
|
|
|
|
|
handle = safemalloc(sizeof *handle);
|
|
|
|
*handle = fd;
|
|
|
|
return handle;
|
|
|
|
|
|
|
|
out:
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2003-01-14 19:09:24 +00:00
|
|
|
char *read_setting_s(void *handle, char const *key, char *buffer, int buflen) {
|
2002-11-19 02:13:46 +00:00
|
|
|
int fd;
|
|
|
|
Handle h;
|
2002-12-12 23:55:51 +00:00
|
|
|
size_t len;
|
2002-11-19 02:13:46 +00:00
|
|
|
|
|
|
|
if (handle == NULL) goto out;
|
|
|
|
fd = *(int *)handle;
|
|
|
|
UseResFile(fd);
|
|
|
|
if (ResError() != noErr) goto out;
|
|
|
|
h = get1namedresource(FOUR_CHAR_CODE('TEXT'), key);
|
|
|
|
if (h == NULL) goto out;
|
|
|
|
|
2002-12-12 23:55:51 +00:00
|
|
|
len = GetHandleSize(h);
|
|
|
|
if (len + 1 > buflen) goto out;
|
|
|
|
memcpy(buffer, *h, len);
|
|
|
|
buffer[len] = '\0';
|
|
|
|
|
2002-11-19 02:13:46 +00:00
|
|
|
ReleaseResource(h);
|
|
|
|
if (ResError() != noErr) goto out;
|
|
|
|
return buffer;
|
|
|
|
|
|
|
|
out:
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2003-01-14 19:09:24 +00:00
|
|
|
int read_setting_i(void *handle, char const *key, int defvalue) {
|
2002-11-19 02:13:46 +00:00
|
|
|
int fd;
|
|
|
|
Handle h;
|
|
|
|
int value;
|
|
|
|
|
|
|
|
if (handle == NULL) goto out;
|
|
|
|
fd = *(int *)handle;
|
|
|
|
UseResFile(fd);
|
|
|
|
if (ResError() != noErr) goto out;
|
|
|
|
h = get1namedresource(FOUR_CHAR_CODE('Int '), key);
|
|
|
|
if (h == NULL) goto out;
|
|
|
|
value = *(int *)*h;
|
|
|
|
ReleaseResource(h);
|
|
|
|
if (ResError() != noErr) goto out;
|
|
|
|
return value;
|
|
|
|
|
|
|
|
out:
|
|
|
|
return defvalue;
|
|
|
|
}
|
|
|
|
|
|
|
|
void close_settings_r(void *handle) {
|
|
|
|
int fd;
|
|
|
|
|
|
|
|
if (handle == NULL) return;
|
|
|
|
fd = *(int *)handle;
|
|
|
|
CloseResFile(fd);
|
|
|
|
if (ResError() != noErr)
|
|
|
|
fatalbox("Close of saved session failed (%d)", ResError());
|
|
|
|
safefree(handle);
|
|
|
|
}
|
|
|
|
|
2003-01-14 19:09:24 +00:00
|
|
|
void del_settings(char const *sessionname) {
|
2002-11-19 02:13:46 +00:00
|
|
|
OSErr error;
|
|
|
|
FSSpec sessfile;
|
|
|
|
short sessVRefNum;
|
|
|
|
long sessDirID;
|
|
|
|
Str255 psessionname;
|
|
|
|
|
|
|
|
error = get_session_dir(kDontCreateFolder, &sessVRefNum, &sessDirID);
|
|
|
|
|
|
|
|
c2pstrcpy(psessionname, sessionname);
|
|
|
|
error = FSMakeFSSpec(sessVRefNum, sessDirID, psessionname, &sessfile);
|
|
|
|
if (error != noErr) goto out;
|
|
|
|
|
|
|
|
error = FSpDelete(&sessfile);
|
|
|
|
return;
|
|
|
|
out:
|
|
|
|
fatalbox("Delete session failed (%d)", error);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct enum_settings_state {
|
|
|
|
short vRefNum;
|
|
|
|
long dirID;
|
|
|
|
int index;
|
|
|
|
};
|
|
|
|
|
|
|
|
void *enum_settings_start(void) {
|
|
|
|
OSErr error;
|
|
|
|
struct enum_settings_state *state;
|
|
|
|
|
|
|
|
state = safemalloc(sizeof(*state));
|
|
|
|
error = get_session_dir(kDontCreateFolder, &state->vRefNum, &state->dirID);
|
|
|
|
if (error != noErr) {
|
|
|
|
safefree(state);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
state->index = 1;
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *enum_settings_next(void *handle, char *buffer, int buflen) {
|
|
|
|
struct enum_settings_state *e = handle;
|
|
|
|
CInfoPBRec pb;
|
|
|
|
OSErr error = noErr;
|
|
|
|
Str255 name;
|
|
|
|
|
|
|
|
if (e == NULL) return NULL;
|
|
|
|
do {
|
|
|
|
pb.hFileInfo.ioNamePtr = name;
|
|
|
|
pb.hFileInfo.ioVRefNum = e->vRefNum;
|
|
|
|
pb.hFileInfo.ioDirID = e->dirID;
|
|
|
|
pb.hFileInfo.ioFDirIndex = e->index++;
|
|
|
|
error = PBGetCatInfoSync(&pb);
|
|
|
|
if (error != noErr) return NULL;
|
|
|
|
} while (!((pb.hFileInfo.ioFlAttrib & ioDirMask) == 0 &&
|
|
|
|
pb.hFileInfo.ioFlFndrInfo.fdCreator == PUTTY_CREATOR &&
|
|
|
|
pb.hFileInfo.ioFlFndrInfo.fdType == SESS_TYPE &&
|
|
|
|
name[0] < buflen));
|
|
|
|
|
|
|
|
p2cstrcpy(buffer, name);
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
void enum_settings_finish(void *handle) {
|
|
|
|
|
|
|
|
safefree(handle);
|
|
|
|
}
|
|
|
|
|
2003-01-08 22:46:12 +00:00
|
|
|
#define SEED_SIZE 512
|
|
|
|
|
|
|
|
void read_random_seed(noise_consumer_t consumer)
|
|
|
|
{
|
|
|
|
short puttyVRefNum;
|
|
|
|
long puttyDirID;
|
|
|
|
OSErr error;
|
|
|
|
char buf[SEED_SIZE];
|
|
|
|
short refnum;
|
|
|
|
long count = SEED_SIZE;
|
|
|
|
|
|
|
|
if (get_putty_dir(kDontCreateFolder, &puttyVRefNum, &puttyDirID) != noErr)
|
|
|
|
return;
|
|
|
|
if (HOpenDF(puttyVRefNum, puttyDirID, "\pPuTTY Random Seed", fsRdPerm,
|
|
|
|
&refnum) != noErr)
|
|
|
|
return;
|
|
|
|
error = FSRead(refnum, &count, buf);
|
|
|
|
if (error != noErr && error != eofErr)
|
|
|
|
return;
|
|
|
|
(*consumer)(buf, count);
|
|
|
|
FSClose(refnum);
|
|
|
|
}
|
|
|
|
|
2003-01-18 12:03:28 +00:00
|
|
|
/*
|
|
|
|
* We don't bother with the usual FSpExchangeFiles dance here because
|
|
|
|
* it doesn't really matter if the old random seed gets lost.
|
|
|
|
*/
|
2003-01-08 22:46:12 +00:00
|
|
|
void write_random_seed(void *data, int len)
|
|
|
|
{
|
2003-01-18 12:03:28 +00:00
|
|
|
short puttyVRefNum;
|
|
|
|
long puttyDirID;
|
2003-01-08 22:46:12 +00:00
|
|
|
OSErr error;
|
2003-01-18 12:03:28 +00:00
|
|
|
FSSpec dstfile;
|
2003-01-08 22:46:12 +00:00
|
|
|
short refnum;
|
|
|
|
long count = len;
|
|
|
|
|
|
|
|
if (get_putty_dir(kCreateFolder, &puttyVRefNum, &puttyDirID) != noErr)
|
|
|
|
return;
|
|
|
|
|
|
|
|
error = FSMakeFSSpec(puttyVRefNum, puttyDirID, "\pPuTTY Random Seed",
|
|
|
|
&dstfile);
|
2003-01-18 12:03:28 +00:00
|
|
|
if (error == fnfErr)
|
|
|
|
error = FSpCreate(&dstfile, PUTTY_CREATOR, SEED_TYPE, smRoman);
|
2003-01-08 22:46:12 +00:00
|
|
|
if (error != noErr) return;
|
|
|
|
|
2003-01-18 12:03:28 +00:00
|
|
|
if (FSpOpenDF(&dstfile, fsWrPerm, &refnum) != noErr) return;
|
2003-01-08 22:46:12 +00:00
|
|
|
|
2003-01-18 12:03:28 +00:00
|
|
|
FSWrite(refnum, &count, data);
|
2003-01-08 22:46:12 +00:00
|
|
|
|
|
|
|
FSClose(refnum);
|
2003-01-18 12:03:28 +00:00
|
|
|
return;
|
2003-01-08 22:46:12 +00:00
|
|
|
}
|
2002-11-19 02:13:46 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Emacs magic:
|
|
|
|
* Local Variables:
|
|
|
|
* c-file-style: "simon"
|
|
|
|
* End:
|
|
|
|
*/
|