2 * settings.c: read and write saved sessions.
11 * Tables of string <-> enum value mappings
13 struct keyval
{ char *s
; int v
; };
15 static const struct keyval ciphernames
[] = {
16 { "aes", CIPHER_AES
},
17 { "blowfish", CIPHER_BLOWFISH
},
18 { "3des", CIPHER_3DES
},
19 { "WARN", CIPHER_WARN
},
23 static void gpps(void *handle
, char *name
, char *def
, char *val
, int len
)
25 if (!read_setting_s(handle
, name
, val
, len
)) {
26 strncpy(val
, def
, len
);
31 static void gppi(void *handle
, char *name
, int def
, int *i
)
33 *i
= read_setting_i(handle
, name
, def
);
36 static int key2val(const struct keyval
*mapping
, int nmaps
, char *key
)
39 for (i
= 0; i
< nmaps
; i
++)
40 if (!strcmp(mapping
[i
].s
, key
)) return mapping
[i
].v
;
44 static const char *val2key(const struct keyval
*mapping
, int nmaps
, int val
)
47 for (i
= 0; i
< nmaps
; i
++)
48 if (mapping
[i
].v
== val
) return mapping
[i
].s
;
53 * Helper function to parse a comma-separated list of strings into
54 * a preference list array of values. Any missing values are added
55 * to the end and duplicates are weeded.
56 * XXX: assumes vals in 'mapping' are small +ve integers
58 static void gprefs(void *sesskey
, char *name
, char *def
,
59 const struct keyval
*mapping
, int nvals
,
64 unsigned long seen
= 0; /* bitmap for weeding dups etc */
65 gpps(sesskey
, name
, def
, commalist
, sizeof(commalist
));
67 /* Grotty parsing of commalist. */
72 key
= strtok(n
==0 ? commalist
: NULL
, ","); /* sorry */
74 if (((v
= key2val(mapping
, nvals
, key
)) != -1) &&
81 /* Add any missing values (backward compatibility ect). */
84 for (i
= 0; i
< nvals
; i
++) {
85 if (!(seen
& 1<<mapping
[i
].v
)) {
86 array
[n
] = mapping
[i
].v
;
94 * Write out a preference list.
96 static void wprefs(void *sesskey
, char *name
,
97 const struct keyval
*mapping
, int nvals
,
100 char buf
[80] = ""; /* XXX assumed big enough */
101 int l
= sizeof(buf
)-1, i
;
103 for (i
= 0; l
> 0 && i
< nvals
; i
++) {
104 const char *s
= val2key(mapping
, nvals
, array
[i
]);
108 strncat(buf
, ",", l
);
115 write_setting_s(sesskey
, name
, buf
);
118 void save_settings(char *section
, int do_host
, Config
* cfg
)
124 sesskey
= open_settings_w(section
);
128 write_setting_i(sesskey
, "Present", 1);
130 write_setting_s(sesskey
, "HostName", cfg
->host
);
131 write_setting_s(sesskey
, "LogFileName", cfg
->logfilename
);
132 write_setting_i(sesskey
, "LogType", cfg
->logtype
);
133 write_setting_i(sesskey
, "LogFileClash", cfg
->logxfovr
);
136 for (i
= 0; backends
[i
].name
!= NULL
; i
++)
137 if (backends
[i
].protocol
== cfg
->protocol
) {
138 p
= backends
[i
].name
;
141 write_setting_s(sesskey
, "Protocol", p
);
142 write_setting_i(sesskey
, "PortNumber", cfg
->port
);
143 write_setting_i(sesskey
, "CloseOnExit", cfg
->close_on_exit
);
144 write_setting_i(sesskey
, "WarnOnClose", !!cfg
->warn_on_close
);
145 write_setting_i(sesskey
, "PingInterval", cfg
->ping_interval
/ 60); /* minutes */
146 write_setting_i(sesskey
, "PingIntervalSecs", cfg
->ping_interval
% 60); /* seconds */
147 write_setting_i(sesskey
, "TCPNoDelay", cfg
->tcp_nodelay
);
148 write_setting_s(sesskey
, "TerminalType", cfg
->termtype
);
149 write_setting_s(sesskey
, "TerminalSpeed", cfg
->termspeed
);
152 write_setting_s(sesskey
, "ProxyExcludeList", cfg
->proxy_exclude_list
);
153 write_setting_i(sesskey
, "ProxyType", cfg
->proxy_type
);
154 write_setting_s(sesskey
, "ProxyHost", cfg
->proxy_host
);
155 write_setting_i(sesskey
, "ProxyPort", cfg
->proxy_port
);
156 write_setting_s(sesskey
, "ProxyUsername", cfg
->proxy_username
);
157 write_setting_s(sesskey
, "ProxyPassword", cfg
->proxy_password
);
158 write_setting_s(sesskey
, "ProxyTelnetCommand", cfg
->proxy_telnet_command
);
159 write_setting_i(sesskey
, "ProxySOCKSVersion", cfg
->proxy_socks_version
);
162 char buf
[2 * sizeof(cfg
->environmt
)], *p
, *q
;
168 if (c
== '=' || c
== ',' || c
== '\\')
178 write_setting_s(sesskey
, "Environment", buf
);
180 write_setting_s(sesskey
, "UserName", cfg
->username
);
181 write_setting_s(sesskey
, "LocalUserName", cfg
->localusername
);
182 write_setting_i(sesskey
, "NoPTY", cfg
->nopty
);
183 write_setting_i(sesskey
, "Compression", cfg
->compression
);
184 write_setting_i(sesskey
, "AgentFwd", cfg
->agentfwd
);
185 write_setting_i(sesskey
, "ChangeUsername", cfg
->change_username
);
186 wprefs(sesskey
, "Cipher", ciphernames
, CIPHER_MAX
,
187 cfg
->ssh_cipherlist
);
188 write_setting_i(sesskey
, "AuthTIS", cfg
->try_tis_auth
);
189 write_setting_i(sesskey
, "AuthKI", cfg
->try_ki_auth
);
190 write_setting_i(sesskey
, "SshProt", cfg
->sshprot
);
191 write_setting_i(sesskey
, "SSH2DES", cfg
->ssh2_des_cbc
);
192 write_setting_s(sesskey
, "PublicKeyFile", cfg
->keyfile
);
193 write_setting_s(sesskey
, "RemoteCommand", cfg
->remote_cmd
);
194 write_setting_i(sesskey
, "RFCEnviron", cfg
->rfc_environ
);
195 write_setting_i(sesskey
, "PassiveTelnet", cfg
->passive_telnet
);
196 write_setting_i(sesskey
, "BackspaceIsDelete", cfg
->bksp_is_delete
);
197 write_setting_i(sesskey
, "RXVTHomeEnd", cfg
->rxvt_homeend
);
198 write_setting_i(sesskey
, "LinuxFunctionKeys", cfg
->funky_type
);
199 write_setting_i(sesskey
, "NoApplicationKeys", cfg
->no_applic_k
);
200 write_setting_i(sesskey
, "NoApplicationCursors", cfg
->no_applic_c
);
201 write_setting_i(sesskey
, "NoMouseReporting", cfg
->no_mouse_rep
);
202 write_setting_i(sesskey
, "NoRemoteResize", cfg
->no_remote_resize
);
203 write_setting_i(sesskey
, "NoAltScreen", cfg
->no_alt_screen
);
204 write_setting_i(sesskey
, "NoRemoteWinTitle", cfg
->no_remote_wintitle
);
205 write_setting_i(sesskey
, "NoDBackspace", cfg
->no_dbackspace
);
206 write_setting_i(sesskey
, "NoRemoteCharset", cfg
->no_remote_charset
);
207 write_setting_i(sesskey
, "ApplicationCursorKeys", cfg
->app_cursor
);
208 write_setting_i(sesskey
, "ApplicationKeypad", cfg
->app_keypad
);
209 write_setting_i(sesskey
, "NetHackKeypad", cfg
->nethack_keypad
);
210 write_setting_i(sesskey
, "AltF4", cfg
->alt_f4
);
211 write_setting_i(sesskey
, "AltSpace", cfg
->alt_space
);
212 write_setting_i(sesskey
, "AltOnly", cfg
->alt_only
);
213 write_setting_i(sesskey
, "ComposeKey", cfg
->compose_key
);
214 write_setting_i(sesskey
, "CtrlAltKeys", cfg
->ctrlaltkeys
);
215 write_setting_i(sesskey
, "TelnetKey", cfg
->telnet_keyboard
);
216 write_setting_i(sesskey
, "TelnetRet", cfg
->telnet_newline
);
217 write_setting_i(sesskey
, "LocalEcho", cfg
->localecho
);
218 write_setting_i(sesskey
, "LocalEdit", cfg
->localedit
);
219 write_setting_s(sesskey
, "Answerback", cfg
->answerback
);
220 write_setting_i(sesskey
, "AlwaysOnTop", cfg
->alwaysontop
);
221 write_setting_i(sesskey
, "FullScreenOnAltEnter", cfg
->fullscreenonaltenter
);
222 write_setting_i(sesskey
, "HideMousePtr", cfg
->hide_mouseptr
);
223 write_setting_i(sesskey
, "SunkenEdge", cfg
->sunken_edge
);
224 write_setting_i(sesskey
, "WindowBorder", cfg
->window_border
);
225 write_setting_i(sesskey
, "CurType", cfg
->cursor_type
);
226 write_setting_i(sesskey
, "BlinkCur", cfg
->blink_cur
);
227 write_setting_i(sesskey
, "Beep", cfg
->beep
);
228 write_setting_i(sesskey
, "BeepInd", cfg
->beep_ind
);
229 write_setting_s(sesskey
, "BellWaveFile", cfg
->bell_wavefile
);
230 write_setting_i(sesskey
, "BellOverload", cfg
->bellovl
);
231 write_setting_i(sesskey
, "BellOverloadN", cfg
->bellovl_n
);
232 write_setting_i(sesskey
, "BellOverloadT", cfg
->bellovl_t
);
233 write_setting_i(sesskey
, "BellOverloadS", cfg
->bellovl_s
);
234 write_setting_i(sesskey
, "ScrollbackLines", cfg
->savelines
);
235 write_setting_i(sesskey
, "DECOriginMode", cfg
->dec_om
);
236 write_setting_i(sesskey
, "AutoWrapMode", cfg
->wrap_mode
);
237 write_setting_i(sesskey
, "LFImpliesCR", cfg
->lfhascr
);
238 write_setting_i(sesskey
, "WinNameAlways", cfg
->win_name_always
);
239 write_setting_s(sesskey
, "WinTitle", cfg
->wintitle
);
240 write_setting_i(sesskey
, "TermWidth", cfg
->width
);
241 write_setting_i(sesskey
, "TermHeight", cfg
->height
);
242 write_setting_s(sesskey
, "Font", cfg
->font
);
243 write_setting_i(sesskey
, "FontIsBold", cfg
->fontisbold
);
244 write_setting_i(sesskey
, "FontCharSet", cfg
->fontcharset
);
245 write_setting_i(sesskey
, "FontHeight", cfg
->fontheight
);
246 write_setting_i(sesskey
, "FontVTMode", cfg
->vtmode
);
247 write_setting_i(sesskey
, "TryPalette", cfg
->try_palette
);
248 write_setting_i(sesskey
, "BoldAsColour", cfg
->bold_colour
);
249 for (i
= 0; i
< 22; i
++) {
250 char buf
[20], buf2
[30];
251 sprintf(buf
, "Colour%d", i
);
252 sprintf(buf2
, "%d,%d,%d", cfg
->colours
[i
][0],
253 cfg
->colours
[i
][1], cfg
->colours
[i
][2]);
254 write_setting_s(sesskey
, buf
, buf2
);
256 write_setting_i(sesskey
, "RawCNP", cfg
->rawcnp
);
257 write_setting_i(sesskey
, "PasteRTF", cfg
->rtf_paste
);
258 write_setting_i(sesskey
, "MouseIsXterm", cfg
->mouse_is_xterm
);
259 write_setting_i(sesskey
, "RectSelect", cfg
->rect_select
);
260 write_setting_i(sesskey
, "MouseOverride", cfg
->mouse_override
);
261 for (i
= 0; i
< 256; i
+= 32) {
262 char buf
[20], buf2
[256];
264 sprintf(buf
, "Wordness%d", i
);
266 for (j
= i
; j
< i
+ 32; j
++) {
267 sprintf(buf2
+ strlen(buf2
), "%s%d",
268 (*buf2 ?
"," : ""), cfg
->wordness
[j
]);
270 write_setting_s(sesskey
, buf
, buf2
);
272 write_setting_s(sesskey
, "LineCodePage", cfg
->line_codepage
);
273 write_setting_s(sesskey
, "Printer", cfg
->printer
);
274 write_setting_i(sesskey
, "CapsLockCyr", cfg
->xlat_capslockcyr
);
275 write_setting_i(sesskey
, "ScrollBar", cfg
->scrollbar
);
276 write_setting_i(sesskey
, "ScrollBarFullScreen", cfg
->scrollbar_in_fullscreen
);
277 write_setting_i(sesskey
, "ScrollOnKey", cfg
->scroll_on_key
);
278 write_setting_i(sesskey
, "ScrollOnDisp", cfg
->scroll_on_disp
);
279 write_setting_i(sesskey
, "LockSize", cfg
->resize_action
);
280 write_setting_i(sesskey
, "BCE", cfg
->bce
);
281 write_setting_i(sesskey
, "BlinkText", cfg
->blinktext
);
282 write_setting_i(sesskey
, "X11Forward", cfg
->x11_forward
);
283 write_setting_s(sesskey
, "X11Display", cfg
->x11_display
);
284 write_setting_i(sesskey
, "LocalPortAcceptAll", cfg
->lport_acceptall
);
285 write_setting_i(sesskey
, "RemotePortAcceptAll", cfg
->rport_acceptall
);
287 char buf
[2 * sizeof(cfg
->portfwd
)], *p
, *q
;
293 if (c
== '=' || c
== ',' || c
== '\\')
303 write_setting_s(sesskey
, "PortForwardings", buf
);
305 write_setting_i(sesskey
, "BugIgnore1", cfg
->sshbug_ignore1
);
306 write_setting_i(sesskey
, "BugPlainPW1", cfg
->sshbug_plainpw1
);
307 write_setting_i(sesskey
, "BugRSA1", cfg
->sshbug_rsa1
);
308 write_setting_i(sesskey
, "BugHMAC2", cfg
->sshbug_hmac2
);
309 write_setting_i(sesskey
, "BugDeriveKey2", cfg
->sshbug_derivekey2
);
310 write_setting_i(sesskey
, "BugRSAPad2", cfg
->sshbug_rsapad2
);
311 write_setting_i(sesskey
, "BugDHGEx2", cfg
->sshbug_dhgex2
);
312 write_setting_i(sesskey
, "StampUtmp", cfg
->stamp_utmp
);
313 write_setting_i(sesskey
, "LoginShell", cfg
->login_shell
);
314 write_setting_i(sesskey
, "ScrollbarOnLeft", cfg
->scrollbar_on_left
);
315 write_setting_s(sesskey
, "BoldFont", cfg
->boldfont
);
316 write_setting_i(sesskey
, "ShadowBoldOffset", cfg
->shadowboldoffset
);
317 close_settings_w(sesskey
);
320 void load_settings(char *section
, int do_host
, Config
* cfg
)
326 sesskey
= open_settings_r(section
);
328 cfg
->ssh_subsys
= 0; /* FIXME: load this properly */
329 cfg
->remote_cmd_ptr
= cfg
->remote_cmd
;
330 cfg
->remote_cmd_ptr2
= NULL
;
333 gpps(sesskey
, "HostName", "", cfg
->host
, sizeof(cfg
->host
));
335 cfg
->host
[0] = '\0'; /* blank hostname */
337 gpps(sesskey
, "LogFileName", "putty.log",
338 cfg
->logfilename
, sizeof(cfg
->logfilename
));
339 gppi(sesskey
, "LogType", 0, &cfg
->logtype
);
340 gppi(sesskey
, "LogFileClash", LGXF_ASK
, &cfg
->logxfovr
);
342 gpps(sesskey
, "Protocol", "default", prot
, 10);
343 cfg
->protocol
= default_protocol
;
344 cfg
->port
= default_port
;
345 for (i
= 0; backends
[i
].name
!= NULL
; i
++)
346 if (!strcmp(prot
, backends
[i
].name
)) {
347 cfg
->protocol
= backends
[i
].protocol
;
348 gppi(sesskey
, "PortNumber", default_port
, &cfg
->port
);
353 * CloseOnExit defaults to closing only on a clean exit - but
354 * unfortunately not on Unix (pterm). On Unix, the exit code of
355 * a shell is the last exit code of one of its child processes,
356 * even if it's an interactive shell - so some pterms will
357 * close and some will not for no particularly good reason. The
358 * mode is still useful for specialist purposes (running a
359 * single command in its own pterm), but I don't think it's a
360 * sane default, unfortunately.
362 gppi(sesskey
, "CloseOnExit",
368 &cfg
->close_on_exit
);
369 gppi(sesskey
, "WarnOnClose", 1, &cfg
->warn_on_close
);
371 /* This is two values for backward compatibility with 0.50/0.51 */
372 int pingmin
, pingsec
;
373 gppi(sesskey
, "PingInterval", 0, &pingmin
);
374 gppi(sesskey
, "PingIntervalSecs", 0, &pingsec
);
375 cfg
->ping_interval
= pingmin
* 60 + pingsec
;
377 gppi(sesskey
, "TCPNoDelay", 1, &cfg
->tcp_nodelay
);
378 gpps(sesskey
, "TerminalType", "xterm", cfg
->termtype
,
379 sizeof(cfg
->termtype
));
380 gpps(sesskey
, "TerminalSpeed", "38400,38400", cfg
->termspeed
,
381 sizeof(cfg
->termspeed
));
384 gpps(sesskey
, "ProxyExcludeList", "", cfg
->proxy_exclude_list
,
385 sizeof(cfg
->proxy_exclude_list
));
386 gppi(sesskey
, "ProxyType", PROXY_NONE
, &i
); cfg
->proxy_type
= i
;
387 gpps(sesskey
, "ProxyHost", "proxy", cfg
->proxy_host
,
388 sizeof(cfg
->proxy_host
));
389 gppi(sesskey
, "ProxyPort", 80, &cfg
->proxy_port
);
390 gpps(sesskey
, "ProxyUsername", "", cfg
->proxy_username
,
391 sizeof(cfg
->proxy_username
));
392 gpps(sesskey
, "ProxyPassword", "", cfg
->proxy_password
,
393 sizeof(cfg
->proxy_password
));
394 gpps(sesskey
, "ProxyTelnetCommand", "connect %host %port\\n",
395 cfg
->proxy_telnet_command
, sizeof(cfg
->proxy_telnet_command
));
396 gppi(sesskey
, "ProxySOCKSVersion", 5, &cfg
->proxy_socks_version
);
399 char buf
[2 * sizeof(cfg
->environmt
)], *p
, *q
;
400 gpps(sesskey
, "Environment", "", buf
, sizeof(buf
));
404 while (*p
&& *p
!= ',') {
418 gpps(sesskey
, "UserName", "", cfg
->username
, sizeof(cfg
->username
));
419 gpps(sesskey
, "LocalUserName", "", cfg
->localusername
,
420 sizeof(cfg
->localusername
));
421 gppi(sesskey
, "NoPTY", 0, &cfg
->nopty
);
422 gppi(sesskey
, "Compression", 0, &cfg
->compression
);
423 gppi(sesskey
, "AgentFwd", 0, &cfg
->agentfwd
);
424 gppi(sesskey
, "ChangeUsername", 0, &cfg
->change_username
);
425 gprefs(sesskey
, "Cipher", "\0",
426 ciphernames
, CIPHER_MAX
, cfg
->ssh_cipherlist
);
427 gppi(sesskey
, "SshProt", 2, &cfg
->sshprot
);
428 gppi(sesskey
, "SSH2DES", 0, &cfg
->ssh2_des_cbc
);
429 gppi(sesskey
, "AuthTIS", 0, &cfg
->try_tis_auth
);
430 gppi(sesskey
, "AuthKI", 1, &cfg
->try_ki_auth
);
431 gpps(sesskey
, "PublicKeyFile", "", cfg
->keyfile
, sizeof(cfg
->keyfile
));
432 gpps(sesskey
, "RemoteCommand", "", cfg
->remote_cmd
,
433 sizeof(cfg
->remote_cmd
));
434 gppi(sesskey
, "RFCEnviron", 0, &cfg
->rfc_environ
);
435 gppi(sesskey
, "PassiveTelnet", 0, &cfg
->passive_telnet
);
436 gppi(sesskey
, "BackspaceIsDelete", 1, &cfg
->bksp_is_delete
);
437 gppi(sesskey
, "RXVTHomeEnd", 0, &cfg
->rxvt_homeend
);
438 gppi(sesskey
, "LinuxFunctionKeys", 0, &cfg
->funky_type
);
439 gppi(sesskey
, "NoApplicationKeys", 0, &cfg
->no_applic_k
);
440 gppi(sesskey
, "NoApplicationCursors", 0, &cfg
->no_applic_c
);
441 gppi(sesskey
, "NoMouseReporting", 0, &cfg
->no_mouse_rep
);
442 gppi(sesskey
, "NoRemoteResize", 0, &cfg
->no_remote_resize
);
443 gppi(sesskey
, "NoAltScreen", 0, &cfg
->no_alt_screen
);
444 gppi(sesskey
, "NoRemoteWinTitle", 0, &cfg
->no_remote_wintitle
);
445 gppi(sesskey
, "NoDBackspace", 0, &cfg
->no_dbackspace
);
446 gppi(sesskey
, "NoRemoteCharset", 0, &cfg
->no_remote_charset
);
447 gppi(sesskey
, "ApplicationCursorKeys", 0, &cfg
->app_cursor
);
448 gppi(sesskey
, "ApplicationKeypad", 0, &cfg
->app_keypad
);
449 gppi(sesskey
, "NetHackKeypad", 0, &cfg
->nethack_keypad
);
450 gppi(sesskey
, "AltF4", 1, &cfg
->alt_f4
);
451 gppi(sesskey
, "AltSpace", 0, &cfg
->alt_space
);
452 gppi(sesskey
, "AltOnly", 0, &cfg
->alt_only
);
453 gppi(sesskey
, "ComposeKey", 0, &cfg
->compose_key
);
454 gppi(sesskey
, "CtrlAltKeys", 1, &cfg
->ctrlaltkeys
);
455 gppi(sesskey
, "TelnetKey", 0, &cfg
->telnet_keyboard
);
456 gppi(sesskey
, "TelnetRet", 1, &cfg
->telnet_newline
);
457 gppi(sesskey
, "LocalEcho", LD_BACKEND
, &cfg
->localecho
);
458 gppi(sesskey
, "LocalEdit", LD_BACKEND
, &cfg
->localedit
);
459 gpps(sesskey
, "Answerback", "PuTTY", cfg
->answerback
,
460 sizeof(cfg
->answerback
));
461 gppi(sesskey
, "AlwaysOnTop", 0, &cfg
->alwaysontop
);
462 gppi(sesskey
, "FullScreenOnAltEnter", 0, &cfg
->fullscreenonaltenter
);
463 gppi(sesskey
, "HideMousePtr", 0, &cfg
->hide_mouseptr
);
464 gppi(sesskey
, "SunkenEdge", 0, &cfg
->sunken_edge
);
465 gppi(sesskey
, "WindowBorder", 1, &cfg
->window_border
);
466 gppi(sesskey
, "CurType", 0, &cfg
->cursor_type
);
467 gppi(sesskey
, "BlinkCur", 0, &cfg
->blink_cur
);
468 /* pedantic compiler tells me I can't use &cfg->beep as an int * :-) */
469 gppi(sesskey
, "Beep", 1, &i
); cfg
->beep
= i
;
470 gppi(sesskey
, "BeepInd", 0, &i
); cfg
->beep_ind
= i
;
471 gpps(sesskey
, "BellWaveFile", "", cfg
->bell_wavefile
,
472 sizeof(cfg
->bell_wavefile
));
473 gppi(sesskey
, "BellOverload", 1, &cfg
->bellovl
);
474 gppi(sesskey
, "BellOverloadN", 5, &cfg
->bellovl_n
);
475 gppi(sesskey
, "BellOverloadT", 2*TICKSPERSEC
, &cfg
->bellovl_t
);
476 gppi(sesskey
, "BellOverloadS", 5*TICKSPERSEC
, &cfg
->bellovl_s
);
477 gppi(sesskey
, "ScrollbackLines", 200, &cfg
->savelines
);
478 gppi(sesskey
, "DECOriginMode", 0, &cfg
->dec_om
);
479 gppi(sesskey
, "AutoWrapMode", 1, &cfg
->wrap_mode
);
480 gppi(sesskey
, "LFImpliesCR", 0, &cfg
->lfhascr
);
481 gppi(sesskey
, "WinNameAlways", 0, &cfg
->win_name_always
);
482 gpps(sesskey
, "WinTitle", "", cfg
->wintitle
, sizeof(cfg
->wintitle
));
483 gppi(sesskey
, "TermWidth", 80, &cfg
->width
);
484 gppi(sesskey
, "TermHeight", 24, &cfg
->height
);
486 gpps(sesskey
, "Font", "Courier New", cfg
->font
, sizeof(cfg
->font
));
487 #elif defined(macintosh)
488 gpps(sesskey
, "Font", "Monaco", cfg
->font
, sizeof(cfg
->font
));
490 gpps(sesskey
, "Font", "fixed", cfg
->font
, sizeof(cfg
->font
));
492 gppi(sesskey
, "FontIsBold", 0, &cfg
->fontisbold
);
494 gppi(sesskey
, "FontCharSet", ANSI_CHARSET
, &cfg
->fontcharset
);
496 gppi(sesskey
, "FontHeight", 10, &cfg
->fontheight
);
498 if (cfg
->fontheight
< 0) {
500 HDC hdc
= GetDC(NULL
);
501 int logpix
= GetDeviceCaps(hdc
, LOGPIXELSY
);
502 ReleaseDC(NULL
, hdc
);
504 oldh
= -cfg
->fontheight
;
505 newh
= MulDiv(oldh
, 72, logpix
) + 1;
506 if (MulDiv(newh
, logpix
, 72) > oldh
)
508 cfg
->fontheight
= newh
;
511 gppi(sesskey
, "FontVTMode", VT_UNICODE
, (int *) &cfg
->vtmode
);
512 gppi(sesskey
, "TryPalette", 0, &cfg
->try_palette
);
513 gppi(sesskey
, "BoldAsColour", 1, &cfg
->bold_colour
);
514 for (i
= 0; i
< 22; i
++) {
515 static char *defaults
[] = {
516 "187,187,187", "255,255,255", "0,0,0", "85,85,85", "0,0,0",
517 "0,255,0", "0,0,0", "85,85,85", "187,0,0", "255,85,85",
518 "0,187,0", "85,255,85", "187,187,0", "255,255,85", "0,0,187",
519 "85,85,255", "187,0,187", "255,85,255", "0,187,187",
520 "85,255,255", "187,187,187", "255,255,255"
522 char buf
[20], buf2
[30];
524 sprintf(buf
, "Colour%d", i
);
525 gpps(sesskey
, buf
, defaults
[i
], buf2
, sizeof(buf2
));
526 if (sscanf(buf2
, "%d,%d,%d", &c0
, &c1
, &c2
) == 3) {
527 cfg
->colours
[i
][0] = c0
;
528 cfg
->colours
[i
][1] = c1
;
529 cfg
->colours
[i
][2] = c2
;
533 /* Non-raw cut and paste of line-drawing chars works badly on the
534 * current Unix stub implementation of the Unicode functions.
535 * So I'm going to temporarily set the default to raw mode so
536 * that the failure mode isn't quite so drastically horrid.
537 * When Unicode comes in, this can all be put right. */
538 gppi(sesskey
, "RawCNP", 1, &cfg
->rawcnp
);
540 gppi(sesskey
, "RawCNP", 0, &cfg
->rawcnp
);
542 gppi(sesskey
, "PasteRTF", 0, &cfg
->rtf_paste
);
543 gppi(sesskey
, "MouseIsXterm", 0, &cfg
->mouse_is_xterm
);
544 gppi(sesskey
, "RectSelect", 0, &cfg
->rect_select
);
545 gppi(sesskey
, "MouseOverride", 1, &cfg
->mouse_override
);
546 for (i
= 0; i
< 256; i
+= 32) {
547 static char *defaults
[] = {
548 "0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0",
549 "0,1,2,1,1,1,1,1,1,1,1,1,1,2,2,2,2,2,2,2,2,2,2,2,2,2,1,1,1,1,1,1",
550 "1,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,1,1,1,1,2",
551 "1,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,1,1,1,1,1",
552 "1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1",
553 "1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1",
554 "2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,1,2,2,2,2,2,2,2,2",
555 "2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,1,2,2,2,2,2,2,2,2"
557 char buf
[20], buf2
[256], *p
;
559 sprintf(buf
, "Wordness%d", i
);
560 gpps(sesskey
, buf
, defaults
[i
/ 32], buf2
, sizeof(buf2
));
562 for (j
= i
; j
< i
+ 32; j
++) {
564 while (*p
&& *p
!= ',')
568 cfg
->wordness
[j
] = atoi(q
);
572 * The empty default for LineCodePage will be converted later
573 * into a plausible default for the locale.
575 gpps(sesskey
, "LineCodePage", "", cfg
->line_codepage
,
576 sizeof(cfg
->line_codepage
));
577 gpps(sesskey
, "Printer", "", cfg
->printer
, sizeof(cfg
->printer
));
578 gppi (sesskey
, "CapsLockCyr", 0, &cfg
->xlat_capslockcyr
);
579 gppi(sesskey
, "ScrollBar", 1, &cfg
->scrollbar
);
580 gppi(sesskey
, "ScrollBarFullScreen", 0, &cfg
->scrollbar_in_fullscreen
);
581 gppi(sesskey
, "ScrollOnKey", 0, &cfg
->scroll_on_key
);
582 gppi(sesskey
, "ScrollOnDisp", 1, &cfg
->scroll_on_disp
);
583 gppi(sesskey
, "LockSize", 0, &i
); cfg
->resize_action
= i
;
584 gppi(sesskey
, "BCE", 1, &cfg
->bce
);
585 gppi(sesskey
, "BlinkText", 0, &cfg
->blinktext
);
586 gppi(sesskey
, "X11Forward", 0, &cfg
->x11_forward
);
587 gpps(sesskey
, "X11Display", "localhost:0", cfg
->x11_display
,
588 sizeof(cfg
->x11_display
));
590 gppi(sesskey
, "LocalPortAcceptAll", 0, &cfg
->lport_acceptall
);
591 gppi(sesskey
, "RemotePortAcceptAll", 0, &cfg
->rport_acceptall
);
593 char buf
[2 * sizeof(cfg
->portfwd
)], *p
, *q
;
594 gpps(sesskey
, "PortForwardings", "", buf
, sizeof(buf
));
598 while (*p
&& *p
!= ',') {
612 gppi(sesskey
, "BugIgnore1", BUG_AUTO
, &i
); cfg
->sshbug_ignore1
= i
;
613 gppi(sesskey
, "BugPlainPW1", BUG_AUTO
, &i
); cfg
->sshbug_plainpw1
= i
;
614 gppi(sesskey
, "BugRSA1", BUG_AUTO
, &i
); cfg
->sshbug_rsa1
= i
;
617 gppi(sesskey
, "BugHMAC2", BUG_AUTO
, &i
); cfg
->sshbug_hmac2
= i
;
618 if (cfg
->sshbug_hmac2
== BUG_AUTO
) {
619 gppi(sesskey
, "BuggyMAC", 0, &i
);
621 cfg
->sshbug_hmac2
= BUG_ON
;
624 gppi(sesskey
, "BugDeriveKey2", BUG_AUTO
, &i
); cfg
->sshbug_derivekey2
= i
;
625 gppi(sesskey
, "BugRSAPad2", BUG_AUTO
, &i
); cfg
->sshbug_rsapad2
= i
;
626 gppi(sesskey
, "BugDHGEx2", BUG_AUTO
, &i
); cfg
->sshbug_dhgex2
= i
;
627 gppi(sesskey
, "StampUtmp", 1, &cfg
->stamp_utmp
);
628 gppi(sesskey
, "LoginShell", 1, &cfg
->login_shell
);
629 gppi(sesskey
, "ScrollbarOnLeft", 0, &cfg
->scrollbar_on_left
);
630 gpps(sesskey
, "BoldFont", "", cfg
->boldfont
, sizeof(cfg
->boldfont
));
631 gppi(sesskey
, "ShadowBoldOffset", 1, &cfg
->shadowboldoffset
);
633 close_settings_r(sesskey
);
636 void do_defaults(char *session
, Config
* cfg
)
639 load_settings(session
, TRUE
, cfg
);
641 load_settings("Default Settings", FALSE
, cfg
);
644 static int sessioncmp(const void *av
, const void *bv
)
646 const char *a
= *(const char *const *) av
;
647 const char *b
= *(const char *const *) bv
;
650 * Alphabetical order, except that "Default Settings" is a
651 * special case and comes first.
653 if (!strcmp(a
, "Default Settings"))
654 return -1; /* a comes first */
655 if (!strcmp(b
, "Default Settings"))
656 return +1; /* b comes first */
658 * FIXME: perhaps we should ignore the first & in determining
661 return strcmp(a
, b
); /* otherwise, compare normally */
664 void get_sesslist(struct sesslist
*list
, int allocate
)
667 int buflen
, bufsize
, i
;
673 buflen
= bufsize
= 0;
675 if ((handle
= enum_settings_start()) != NULL
) {
677 ret
= enum_settings_next(handle
, otherbuf
, sizeof(otherbuf
));
679 int len
= strlen(otherbuf
) + 1;
680 if (bufsize
< buflen
+ len
) {
681 bufsize
= buflen
+ len
+ 2048;
682 list
->buffer
= srealloc(list
->buffer
, bufsize
);
684 strcpy(list
->buffer
+ buflen
, otherbuf
);
685 buflen
+= strlen(list
->buffer
+ buflen
) + 1;
688 enum_settings_finish(handle
);
690 list
->buffer
= srealloc(list
->buffer
, buflen
+ 1);
691 list
->buffer
[buflen
] = '\0';
694 * Now set up the list of sessions. Note that "Default
695 * Settings" must always be claimed to exist, even if it
700 list
->nsessions
= 1; /* "Default Settings" counts as one */
702 if (strcmp(p
, "Default Settings"))
709 list
->sessions
= smalloc((list
->nsessions
+ 1) * sizeof(char *));
710 list
->sessions
[0] = "Default Settings";
714 if (strcmp(p
, "Default Settings"))
715 list
->sessions
[i
++] = p
;
721 qsort(list
->sessions
, i
, sizeof(char *), sessioncmp
);
724 sfree(list
->sessions
);