mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-25 09:12:24 +00:00
61711b9594
Filenames are represented as a FSSpec, which is converted to and from an alias record ('alis' resource) when saving and loading sessions. . It might be an idea to allow in-core Filenames to contain alias records too, so that they can refer to directories that don't exist on the current system, but that requires Filenames to be dynamically allocated, which is likely to be a pain. [originally from svn r2771]
593 lines
15 KiB
C
593 lines
15 KiB
C
/* $Id: macstore.c,v 1.15 2003/02/01 21:44:05 ben Exp $ */
|
|
|
|
/*
|
|
* 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 <stdio.h>
|
|
#include <string.h>
|
|
|
|
#include "putty.h"
|
|
#include "storage.h"
|
|
#include "mac.h"
|
|
#include "macresid.h"
|
|
|
|
|
|
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.
|
|
*/
|
|
|
|
OSErr get_putty_dir(Boolean makeit, short *pVRefNum, long *pDirID)
|
|
{
|
|
OSErr error = noErr;
|
|
short prefVRefNum;
|
|
FSSpec puttydir;
|
|
long prefDirID, puttyDirID;
|
|
|
|
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;
|
|
|
|
*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",
|
|
&sessdir);
|
|
if (error != noErr && error != fnfErr) goto out;
|
|
error = FSpGetDirID(&sessdir, &sessDirID, makeit);
|
|
if (error != noErr) goto out;
|
|
|
|
*pVRefNum = puttyVRefNum;
|
|
*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;
|
|
}
|
|
|
|
/* Copy a resource into the current resource file */
|
|
static OSErr copy_resource(ResType restype, short resid)
|
|
{
|
|
Handle h;
|
|
Str255 resname;
|
|
|
|
h = GetResource(restype, resid);
|
|
if (h != NULL) {
|
|
GetResInfo(h, &resid, &restype, resname);
|
|
DetachResource(h);
|
|
AddResource(h, restype, resid, resname);
|
|
if (ResError() == noErr)
|
|
WriteResource(h);
|
|
}
|
|
return ResError();
|
|
}
|
|
|
|
struct write_settings {
|
|
int fd;
|
|
FSSpec tmpfile;
|
|
FSSpec dstfile;
|
|
};
|
|
|
|
void *open_settings_w(char const *sessionname) {
|
|
short sessVRefNum;
|
|
long sessDirID;
|
|
OSErr error;
|
|
Str255 psessionname;
|
|
FSSpec dstfile;
|
|
|
|
error = get_session_dir(kCreateFolder, &sessVRefNum, &sessDirID);
|
|
if (error != noErr) return NULL;
|
|
|
|
c2pstrcpy(psessionname, sessionname);
|
|
error = FSMakeFSSpec(sessVRefNum, sessDirID, psessionname, &dstfile);
|
|
if (error == fnfErr) {
|
|
FSpCreateResFile(&dstfile, PUTTY_CREATOR, SESS_TYPE, smSystemScript);
|
|
if ((error = ResError()) != noErr) return NULL;
|
|
} else if (error != noErr) return NULL;
|
|
|
|
return open_settings_w_fsp(&dstfile);
|
|
}
|
|
|
|
/*
|
|
* NB: Destination file must exist.
|
|
*/
|
|
void *open_settings_w_fsp(FSSpec *dstfile)
|
|
{
|
|
short tmpVRefNum;
|
|
long tmpDirID;
|
|
struct write_settings *ws;
|
|
OSErr error;
|
|
Str255 tmpname;
|
|
|
|
ws = smalloc(sizeof *ws);
|
|
ws->dstfile = *dstfile;
|
|
|
|
/* Create a temporary file to save to first. */
|
|
error = FindFolder(ws->dstfile.vRefNum, kTemporaryFolderType,
|
|
kCreateFolder, &tmpVRefNum, &tmpDirID);
|
|
if (error != noErr) goto out;
|
|
c2pstrcpy(tmpname, tmpnam(NULL));
|
|
error = FSMakeFSSpec(tmpVRefNum, tmpDirID, tmpname, &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;}
|
|
|
|
/* Set up standard resources. Doesn't matter if these fail. */
|
|
copy_resource('STR ', -16396);
|
|
copy_resource('TMPL', TMPL_Int);
|
|
|
|
return ws;
|
|
|
|
out:
|
|
safefree(ws);
|
|
fatalbox("Failed to open session for write (%d)", error);
|
|
}
|
|
|
|
void write_setting_s(void *handle, char const *key, char const *value) {
|
|
int fd = *(int *)handle;
|
|
Handle h;
|
|
int id;
|
|
OSErr error;
|
|
|
|
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());
|
|
}
|
|
|
|
void write_setting_i(void *handle, char const *key, int value) {
|
|
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);
|
|
}
|
|
|
|
void *open_settings_r(char const *sessionname)
|
|
{
|
|
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;
|
|
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);
|
|
if (fd == 0) {error = ResError(); goto out;}
|
|
|
|
handle = safemalloc(sizeof *handle);
|
|
*handle = fd;
|
|
return handle;
|
|
|
|
out:
|
|
return NULL;
|
|
}
|
|
|
|
char *read_setting_s(void *handle, char const *key, char *buffer, int buflen) {
|
|
int fd;
|
|
Handle h;
|
|
size_t len;
|
|
|
|
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;
|
|
|
|
len = GetHandleSize(h);
|
|
if (len + 1 > buflen) goto out;
|
|
memcpy(buffer, *h, len);
|
|
buffer[len] = '\0';
|
|
|
|
ReleaseResource(h);
|
|
if (ResError() != noErr) goto out;
|
|
return buffer;
|
|
|
|
out:
|
|
return NULL;
|
|
}
|
|
|
|
int read_setting_i(void *handle, char const *key, int defvalue) {
|
|
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;
|
|
}
|
|
|
|
int read_setting_fontspec(void *handle, const char *name, FontSpec *result)
|
|
{
|
|
char *settingname;
|
|
FontSpec ret;
|
|
char tmp[256];
|
|
|
|
if (!read_setting_s(handle, name, tmp, sizeof(tmp)))
|
|
return 0;
|
|
c2pstrcpy(ret.name, tmp);
|
|
settingname = dupcat(name, "Face", NULL);
|
|
ret.face = read_setting_i(handle, settingname, 0);
|
|
sfree(settingname);
|
|
settingname = dupcat(name, "Height", NULL);
|
|
ret.size = read_setting_i(handle, settingname, 0);
|
|
sfree(settingname);
|
|
if (ret.size == 0) return 0;
|
|
*result = ret;
|
|
return 1;
|
|
}
|
|
|
|
void write_setting_fontspec(void *handle, const char *name, FontSpec font)
|
|
{
|
|
char *settingname;
|
|
char tmp[256];
|
|
|
|
p2cstrcpy(tmp, font.name);
|
|
write_setting_s(handle, name, tmp);
|
|
settingname = dupcat(name, "Face", NULL);
|
|
write_setting_i(handle, settingname, font.face);
|
|
sfree(settingname);
|
|
settingname = dupcat(name, "Size", NULL);
|
|
write_setting_i(handle, settingname, font.size);
|
|
sfree(settingname);
|
|
}
|
|
|
|
int read_setting_filename(void *handle, const char *name, Filename *result)
|
|
{
|
|
int fd;
|
|
AliasHandle h;
|
|
Boolean changed;
|
|
OSErr err;
|
|
|
|
if (handle == NULL) goto out;
|
|
fd = *(int *)handle;
|
|
UseResFile(fd);
|
|
if (ResError() != noErr) goto out;
|
|
h = (AliasHandle)get1namedresource(rAliasType, name);
|
|
if (h == NULL) goto out;
|
|
if ((*h)->userType == 'pTTY' && (*h)->aliasSize == sizeof(**h))
|
|
memset(result, 0, sizeof(*result));
|
|
else {
|
|
err = ResolveAlias(NULL, h, &result->fss, &changed);
|
|
if (err != noErr && err != fnfErr) goto out;
|
|
if ((*h)->userType == 'pTTY') {
|
|
long dirid;
|
|
StrFileName fname;
|
|
|
|
/* Tail of record is pascal string contaning leafname */
|
|
if (FSpGetDirID(&result->fss, &dirid, FALSE) != noErr) goto out;
|
|
memcpy(fname, (char *)*h + (*h)->aliasSize,
|
|
GetHandleSize((Handle)h) - (*h)->aliasSize);
|
|
err = FSMakeFSSpec(result->fss.vRefNum, dirid, fname,
|
|
&result->fss);
|
|
if (err != noErr && err != fnfErr) goto out;
|
|
}
|
|
}
|
|
ReleaseResource((Handle)h);
|
|
if (ResError() != noErr) goto out;
|
|
return 1;
|
|
|
|
out:
|
|
return 0;
|
|
}
|
|
|
|
void write_setting_filename(void *handle, const char *name, Filename fn)
|
|
{
|
|
int fd = *(int *)handle;
|
|
AliasHandle h;
|
|
int id;
|
|
OSErr error;
|
|
|
|
UseResFile(fd);
|
|
if (ResError() != noErr)
|
|
fatalbox("Failed to open saved session (%d)", ResError());
|
|
|
|
if (filename_is_null(fn)) {
|
|
/* Generate a special "null" alias */
|
|
h = (AliasHandle)NewHandle(sizeof(**h));
|
|
if (h == NULL)
|
|
fatalbox("Failed to create fake alias");
|
|
(*h)->userType = 'pTTY';
|
|
(*h)->aliasSize = sizeof(**h);
|
|
} else {
|
|
error = NewAlias(NULL, &fn.fss, &h);
|
|
if (error == fnfErr) {
|
|
/*
|
|
* NewAlias can't create an alias for a nonexistent file.
|
|
* Create an alias for the directory, and record the
|
|
* filename as well.
|
|
*/
|
|
FSSpec tmpfss;
|
|
|
|
FSMakeFSSpec(fn.fss.vRefNum, fn.fss.parID, NULL, &tmpfss);
|
|
error = NewAlias(NULL, &tmpfss, &h);
|
|
if (error != noErr)
|
|
fatalbox("Failed to create alias");
|
|
(*h)->userType = 'pTTY';
|
|
SetHandleSize((Handle)h, (*h)->aliasSize + fn.fss.name[0] + 1);
|
|
if (MemError() != noErr)
|
|
fatalbox("Failed to create alias");
|
|
memcpy((char *)*h + (*h)->aliasSize, fn.fss.name,
|
|
fn.fss.name[0] + 1);
|
|
}
|
|
if (error != noErr)
|
|
fatalbox("Failed to create alias");
|
|
}
|
|
/* Put the data in a resource. */
|
|
id = Unique1ID(rAliasType);
|
|
if (ResError() != noErr)
|
|
fatalbox("Failed to get ID for resource %s (%d)", name, ResError());
|
|
addresource((Handle)h, rAliasType, id, name);
|
|
if (ResError() != noErr)
|
|
fatalbox("Failed to add resource %s (%d)", name, ResError());
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
void del_settings(char const *sessionname) {
|
|
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);
|
|
}
|
|
|
|
#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);
|
|
}
|
|
|
|
/*
|
|
* We don't bother with the usual FSpExchangeFiles dance here because
|
|
* it doesn't really matter if the old random seed gets lost.
|
|
*/
|
|
void write_random_seed(void *data, int len)
|
|
{
|
|
short puttyVRefNum;
|
|
long puttyDirID;
|
|
OSErr error;
|
|
FSSpec dstfile;
|
|
short refnum;
|
|
long count = len;
|
|
|
|
if (get_putty_dir(kCreateFolder, &puttyVRefNum, &puttyDirID) != noErr)
|
|
return;
|
|
|
|
error = FSMakeFSSpec(puttyVRefNum, puttyDirID, "\pPuTTY Random Seed",
|
|
&dstfile);
|
|
if (error == fnfErr) {
|
|
/* Set up standard resources */
|
|
FSpCreateResFile(&dstfile, INTERNAL_CREATOR, SEED_TYPE, smRoman);
|
|
refnum = FSpOpenResFile(&dstfile, fsWrPerm);
|
|
if (ResError() == noErr) {
|
|
copy_resource('STR ', -16397);
|
|
CloseResFile(refnum);
|
|
}
|
|
} else if (error != noErr) return;
|
|
|
|
if (FSpOpenDF(&dstfile, fsWrPerm, &refnum) != noErr) return;
|
|
FSWrite(refnum, &count, data);
|
|
FSClose(refnum);
|
|
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* Emacs magic:
|
|
* Local Variables:
|
|
* c-file-style: "simon"
|
|
* End:
|
|
*/
|