sbbs.h 44.3 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
/* sbbs.h */

/* Synchronet class (sbbs_t) definition and exported function prototypes */

/* $Id$ */

/****************************************************************************
 * @format.tab-size 4		(Plain Text/Source Code File Header)			*
 * @format.use-tabs true	(see http://www.synchro.net/ptsc_hdr.html)		*
 *																			*
11
 * Copyright 2012 Rob Swindell - http://www.synchro.net/copyright.html		*
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
 *																			*
 * 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.					*
 * See the GNU General Public License for more details: gpl.txt or			*
 * http://www.fsf.org/copyleft/gpl.html										*
 *																			*
 * Anonymous FTP access to the most recent released source is available at	*
 * ftp://vert.synchro.net, ftp://cvs.synchro.net and ftp://ftp.synchro.net	*
 *																			*
 * Anonymous CVS access to the development source and modification history	*
 * is available at cvs.synchro.net:/cvsroot/sbbs, example:					*
 * cvs -d :pserver:anonymous@cvs.synchro.net:/cvsroot/sbbs login			*
 *     (just hit return, no password is necessary)							*
 * cvs -d :pserver:anonymous@cvs.synchro.net:/cvsroot/sbbs checkout src		*
 *																			*
 * For Synchronet coding style and modification guidelines, see				*
 * http://www.synchro.net/source.html										*
 *																			*
 * You are encouraged to submit any modifications (preferably in Unix diff	*
 * format) via e-mail to mods@synchro.net									*
 *																			*
 * Note: If this box doesn't appear square, then you need to fix your tabs.	*
 ****************************************************************************/

#ifndef _SBBS_H
#define _SBBS_H

/****************************/
/* Standard library headers */
/****************************/

rswindell's avatar
rswindell committed
45
46
47
/***************/
/* OS-specific */
/***************/
48
#if defined(_WIN32)			/* Windows */
49

deuce's avatar
deuce committed
50
51
52
	#define NOCRYPT     /* Stop windows.h from loading wincrypt.h */
                    /* Is windows.h REALLY necessary?!?! */
	#define WIN32_LEAN_AND_MEAN
53
54
55
56
57
58
	#include <io.h>
	#include <share.h>
	#include <windows.h>
	#include <process.h>	/* _beginthread() prototype */
	#include <direct.h>		/* _mkdir() prototype */
	#include <mmsystem.h>	/* SND_ASYNC */
59

60
	#if defined(_DEBUG) && defined(_MSC_VER)
61
62
63
		#include <crtdbg.h> /* Windows debug macros and stuff */
	#endif

64
#if defined(__cplusplus)
65
	extern "C"
66
#endif
deuce's avatar
deuce committed
67
	HINSTANCE hK32;
68

69
#elif defined(__unix__)		/* Unix-variant */
70

71
	#include <unistd.h>		/* close */
72
73
74

#endif

75
76
77
78
79
80
#ifdef _THREAD_SUID_BROKEN
extern int	thread_suid_broken;			/* NPTL is no longer broken */
#else
#define thread_suid_broken FALSE
#endif

rswindell's avatar
rswindell committed
81
82
83
/******************/
/* ANSI C Library */
/******************/
84
85
86
87
#include <time.h>
#include <errno.h>
#include <stdio.h>
#include <ctype.h>
88
#include <fcntl.h>			/* open */
89
90
91
#include <stdarg.h>
#include <stdlib.h>
#include <string.h>
92

93
#ifndef __unix__
94
95
96
97
98

	#include <malloc.h>

#endif

99
100
#include <sys/stat.h>

deuce's avatar
deuce committed
101
102
103
104
105
106
#ifdef __unix__
	#define XP_UNIX
#else
	#define XP_PC
	#define XP_WIN
#endif
rswindell's avatar
rswindell committed
107
108
109

#if defined(JAVASCRIPT)
#include "comio.h"			/* needed for COM_HANDLE definition only */
deuce's avatar
deuce committed
110
111
112
#include <jsversion.h>
#include <jsapi.h>
#define JS_DestroyScript(cx,script)
113

114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
#define JSSTRING_TO_MSTRING(cx, str, ret, lenptr) \
{ \
	size_t			*JSSTSlenptr=lenptr; \
	size_t			JSSTSlen; \
	size_t			JSSTSpos; \
	const jschar	*JSSTSstrval; \
\
	if(JSSTSlenptr==NULL) \
		JSSTSlenptr=&JSSTSlen; \
	(ret)=NULL; \
	if((str) != NULL) { \
		if((JSSTSstrval=JS_GetStringCharsAndLength((cx), (str), JSSTSlenptr))) { \
			if(((ret)=(char *)malloc(*JSSTSlenptr+1))) { \
				for(JSSTSpos=0; JSSTSpos<*JSSTSlenptr; JSSTSpos++) \
					(ret)[JSSTSpos]=(char)JSSTSstrval[JSSTSpos]; \
				(ret)[*JSSTSlenptr]=0; \
			} \
131
			else JS_ReportError(cx, "Error allocating %lu bytes at %s:%d", (*JSSTSlenptr)+1, getfname(__FILE__), __LINE__); \
132
133
134
135
136
137
138
139
140
141
		} \
	} \
}

#define JSVALUE_TO_MSTRING(cx, val, ret, lenptr) \
{ \
	JSString	*JSVTSstr=JS_ValueToString((cx), (val)); \
	JSSTRING_TO_MSTRING((cx), JSVTSstr, (ret), lenptr); \
}

142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
#define JSSTRING_TO_STRBUF(cx, str, ret, bufsize, lenptr) \
{ \
	size_t			*JSSTSlenptr=lenptr; \
	size_t			JSSTSlen; \
	size_t			JSSTSpos; \
	const jschar	*JSSTSstrval; \
\
	if(JSSTSlenptr==NULL) \
		JSSTSlenptr=&JSSTSlen; \
	if(bufsize < 1 || str==NULL) \
		*JSSTSlenptr = 0; \
	else { \
		if((JSSTSstrval=JS_GetStringCharsAndLength((cx), (str), JSSTSlenptr))) { \
			if(*JSSTSlenptr >= bufsize) \
				*JSSTSlenptr = bufsize-1; \
			for(JSSTSpos=0; JSSTSpos<*JSSTSlenptr; JSSTSpos++) \
				(ret)[JSSTSpos]=(char)JSSTSstrval[JSSTSpos]; \
		} \
		else \
			*JSSTSlenptr=0; \
	} \
	(ret)[*JSSTSlenptr]=0; \
}

#define JSVALUE_TO_STRBUF(cx, val, ret, bufsize, lenptr) \
{ \
	JSString	*JSVTSstr=JS_ValueToString((cx), (val)); \
	JSSTRING_TO_STRBUF((cx), JSVTSstr, (ret), (bufsize), lenptr); \
}

#define HANDLE_PENDING(cx) \
	if(JS_IsExceptionPending(cx)) \
		return JS_FALSE;

176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
#define JSSTRING_TO_ASTRING(cx, str, ret, maxsize, lenptr) \
{ \
	size_t			*JSSTSlenptr=lenptr; \
	size_t			JSSTSlen; \
	size_t			JSSTSpos; \
	const jschar	*JSSTSstrval; \
\
	if(JSSTSlenptr==NULL) \
		JSSTSlenptr=&JSSTSlen; \
	(ret)=NULL; \
	if((str) != NULL) { \
		if((JSSTSstrval=JS_GetStringCharsAndLength((cx), (str), JSSTSlenptr))) { \
			if(*JSSTSlenptr >= maxsize) \
				*JSSTSlenptr = maxsize-1; \
			if(((ret)=(char *)alloca(*JSSTSlenptr+1))) { \
				for(JSSTSpos=0; JSSTSpos<*JSSTSlenptr; JSSTSpos++) \
					(ret)[JSSTSpos]=(char)JSSTSstrval[JSSTSpos]; \
				(ret)[*JSSTSlenptr]=0; \
			} \
			else JS_ReportError(cx, "Error allocating %lu bytes on stack at %s:%d", (*JSSTSlenptr)+1, getfname(__FILE__), __LINE__); \
		} \
	} \
}

#define JSVALUE_TO_ASTRING(cx, val, ret, maxsize, lenptr) \
{ \
	JSString	*JSVTSstr=JS_ValueToString((cx), (val)); \
	JSSTRING_TO_ASTRING((cx), JSVTSstr, (ret), (maxsize), (lenptr)); \
}

206
#define JSSTRING_TO_STRING(cx, str, ret, lenptr) \
deuce's avatar
deuce committed
207
{ \
208
	size_t			*JSSTSlenptr=lenptr; \
209
210
211
	size_t			JSSTSlen; \
	size_t			JSSTSpos; \
	const jschar	*JSSTSstrval; \
deuce's avatar
deuce committed
212
\
213
214
	if(JSSTSlenptr==NULL) \
		JSSTSlenptr=&JSSTSlen; \
deuce's avatar
deuce committed
215
216
	(ret)=NULL; \
	if((str) != NULL) { \
217
218
219
		if((JSSTSstrval=JS_GetStringCharsAndLength((cx), (str), JSSTSlenptr))) { \
			if(((ret)=(char *)alloca(*JSSTSlenptr+1))) { \
				for(JSSTSpos=0; JSSTSpos<*JSSTSlenptr; JSSTSpos++) \
rswindell's avatar
rswindell committed
220
					(ret)[JSSTSpos]=(char)JSSTSstrval[JSSTSpos]; \
221
				(ret)[*JSSTSlenptr]=0; \
222
			} \
deuce's avatar
deuce committed
223
224
		} \
	} \
deuce's avatar
deuce committed
225
}
deuce's avatar
deuce committed
226

227
#define JSVALUE_TO_STRING(cx, val, ret, lenptr) \
228
{ \
229
	JSString	*JSVTSstr=JS_ValueToString((cx), (val)); \
230
	JSSTRING_TO_STRING((cx), JSVTSstr, (ret), lenptr); \
231
232
}

rswindell's avatar
rswindell committed
233
234
#endif

235
236
237
238
#ifdef USE_CRYPTLIB
#include <cryptlib.h>
#endif

239
/* xpdev */
240
241
242
#ifndef LINK_LIST_THREADSAFE
 #define LINK_LIST_THREADSAFE
#endif
243
#include "genwrap.h"
rswindell's avatar
rswindell committed
244
#include "semfile.h"
245
#include "netwrap.h"
246
247
#include "dirwrap.h"
#include "filewrap.h"
248
#include "datewrap.h"
249
#include "sockwrap.h"
250
#include "eventwrap.h"
251
#include "link_list.h"
252
#include "msg_queue.h"
253
#include "xpdatetime.h"
254

255
256
257
258
259
260
261
262
/***********************/
/* Synchronet-specific */
/***********************/
#include "startup.h"
#ifdef __cplusplus
	#include "threadwrap.h"	/* pthread_mutex_t */
#endif

263
264
265
266
267
268
269
270
271
#include "smblib.h"
#include "ars_defs.h"
#include "scfgdefs.h"
#include "scfglib.h"
#include "userdat.h"
#include "riodefs.h"
#include "cmdshell.h"
#include "ringbuf.h"    /* RingBuf definition */
#include "client.h"		/* client_t definition */
272
273
#include "crc16.h"
#include "crc32.h"
274
#include "telnet.h"
275
#include "nopen.h"
276
#include "text.h"
277

278
/* Synchronet Node Instance class definition */
279
280
281
282
283
284
#ifdef __cplusplus
class sbbs_t
{

public:

285
	sbbs_t(ushort node_num, SOCKADDR_IN addr, const char* host_name, SOCKET
286
		,scfg_t*, char* text[], client_t* client_info);
287
288
	~sbbs_t();

289
290
	bbs_startup_t*	startup;

291
	bool	init(void);
292
	BOOL	terminated;
293
294
295
296

	client_t client;
	SOCKET	client_socket;
	SOCKET	client_socket_dup;
297
	SOCKADDR_IN	client_addr;
298
	char	client_name[128];
rswindell's avatar
rswindell committed
299
	char	client_ident[128];
300
	DWORD	local_addr;
301
302
303
#ifdef USE_CRYPTLIB
	CRYPT_SESSION	ssh_session;
	bool	ssh_mode;
304
	SOCKET	passthru_socket;
305
306
    bool	passthru_output_thread_running;
    bool	passthru_input_thread_running;
307
#endif
308
309
310
311
312
313
314
315
316
317
318

	scfg_t	cfg;

	int		outchar_esc;		   // track ANSI escape seq output

	int 	rioctl(ushort action); // remote i/o control
	bool	rio_abortable;

    RingBuf	inbuf;
    RingBuf	outbuf;
	HANDLE	input_thread;
319
	pthread_mutex_t	input_thread_mutex;
320
	bool	input_thread_mutex_locked;	// by someone other than the input_thread
321
	pthread_mutex_t	ssh_mutex;
322
323

	int 	outcom(uchar ch); 	   // send character
324
	int 	incom(unsigned long timeout=0);		   // receive character
325

326
	void	spymsg(const char *msg);		// send message to active spies
327

328
	int		putcom(const char *str, size_t len=0);  // Send string
329
330
	void	hangup(void);		   // Hangup modem

331
332
	uchar	telnet_local_option[0x100];
	uchar	telnet_remote_option[0x100];
rswindell's avatar
rswindell committed
333
	void	send_telnet_cmd(uchar cmd, uchar opt);
334
	bool	request_telnet_opt(uchar cmd, uchar opt, unsigned waitforack=0);
335

336
    uchar	telnet_cmd[64];
rswindell's avatar
rswindell committed
337
    uint	telnet_cmdlen;
338
339
	ulong	telnet_mode;
	uchar	telnet_last_rxch;
340
	char	telnet_location[128];
341
	char	terminal[TELNET_TERM_MAXLEN+1];
342
	xpevent_t	telnet_ack_event;
343

344
	time_t	event_time;				// Time of next exclusive event
345
	char*	event_code;				// Internal code of next exclusive event
346
347
	bool	event_thread_running;
    bool	output_thread_running;
348
    bool	input_thread_running;
349

350
351
#ifdef JAVASCRIPT

352
353
354
355
356
357
358
359
	JSRuntime*		js_runtime;
	JSContext*		js_cx;
	JSObject*		js_glob;
	js_callback_t	js_callback;
	long			js_execfile(const char *fname, const char* startup_dir, JSObject* scope=NULL);
	bool			js_init(ulong* stack_frame);
	void			js_cleanup(const char* node);
	void			js_create_user_objects(void);
360
361
362

#endif

363
364
365
366
367
368
	char 	menu_dir[128];	/* Over-ride default menu dir */
	char 	menu_file[128]; /* Over-ride menu file */

	user_t	useron; 		/* User currently online */
	node_t	thisnode;		/* Node information */
	smb_t	smb;			/* Currently open message base */
369
	char	rlogin_name[LEN_ALIAS+1];
370
	char	rlogin_pass[LEN_PASS+1];
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401

	uint	temp_dirnum;

	FILE	*nodefile_fp,
			*node_ext_fp,
			*logfile_fp;

	int 	nodefile;		/* File handle for node.dab */
	int		node_ext;		/* File handle for node.exb */
	int 	inputfile;		/* File handle to use for input */

							/* Batch download queue */
	char 	**batdn_name;	/* Filenames */
	ushort	*batdn_alt; 	/* Alternate path */
	uint 	*batdn_dir, 	/* Directory for each file */
			 batdn_total;	/* Total files */
	long 	*batdn_offset;	/* Offset for data */
	ulong	*batdn_size;	/* Size of file in bytes */
	ulong	*batdn_cdt; 	/* Credit value of file */

							/* Batch upload queue */
	char 	**batup_desc,	/* Description for each file */
			**batup_name;	/* Filenames */
	long	*batup_misc;	/* Miscellaneous bits */
	ushort	*batup_alt; 	/* Alternate path */
	uint 	*batup_dir, 	/* Directory for each file */
			batup_total;	/* Total files */

	/*********************************/
	/* Color Configuration Variables */
	/*********************************/
402
403
	char 	*text[TOTAL_TEXT];			/* Text from ctrl\text.dat */
	char 	*text_sav[TOTAL_TEXT];		/* Text from ctrl\text.dat */
404

405
	char 	dszlog[127];	/* DSZLOG enviornment variable */
406
    int     keybuftop,keybufbot;    /* Keyboard input buffer pointers (for ungetkey) */
407
	char    keybuf[KEY_BUFSIZE];    /* Keyboard input buffer */
408
409

	ushort	node_connection;
410
	char	connection[LEN_MODEM+1];	/* Connection Description */
411
412
413
414
415
416
	ulong	cur_rate;		/* Current Connection (DCE) Rate */
	ulong	cur_cps;		/* Current Average Transfer CPS */
	ulong	dte_rate;		/* Current COM Port (DTE) Rate */
	time_t 	timeout;		/* User inactivity timeout reference */
	ulong 	timeleft_warn;	/* low timeleft warning flag */
	uchar 	curatr; 		/* Current Text Attributes Always */
417
418
	uchar	attr_stack[64];	/* Saved attributes (stack) */
	int 	attr_sp;		/* Attribute stack pointer */
419
420
	long 	lncntr; 		/* Line Counter - for PAUSE */
	long 	tos;			/* Top of Screen */
421
422
	long 	rows;			/* Current number of Rows for User */
	long	cols;			/* Current number of Columns for User */
423
	long	column;			/* Current column counter (for line counter) */
424
425
426
	long 	autoterm;		/* Autodetected terminal type */
	char 	slbuf[SAVE_LINES][LINE_BUFSIZE+1]; /* Saved for redisplay */
	char 	slatr[SAVE_LINES];	/* Starting attribute of each line */
427
428
	char 	slcuratr[SAVE_LINES];	/* Ending attribute of each line */
	int 	slcnt;			/* Number of lines currently saved */
429
430
431
432
433
	char 	lbuf[LINE_BUFSIZE+1];/* Temp storage for each line output */
	int		lbuflen;		/* Number of characters in line buffer */
	char 	latr;			/* Starting attribute of line buffer */
	ulong	console;		/* Defines current Console settings */
	char 	wordwrap[81];	/* Word wrap buffer */
434
	time_t	now,			/* Used to store current time in Unix format */
435
			answertime, 	/* Time call was answered */
436
437
			logontime,		/* Time user logged on */
			starttime,		/* Time stamp to use for time left calcs */
438
			ns_time,		/* File new-scan time */
439
440
441
442
			last_ns_time;	/* Most recent new-file-scan this call */
	uchar 	action;			/* Current action of user */
	long 	online; 		/* Remote/Local or not online */
	long 	sys_status; 	/* System Status */
443
	subscan_t	*subscan;	/* User sub configuration/scan info */
444
445
446
447
448
449
450
451

	ulong	logon_ulb,		/* Upload Bytes This Call */
			logon_dlb,		/* Download Bytes This Call */
			logon_uls,		/* Uploads This Call */
			logon_dls,		/* Downloads This Call */
			logon_posts,	/* Posts This Call */
			logon_emails,	/* Emails This Call */
			logon_fbacks;	/* Feedbacks This Call */
452
	uchar	logon_ml;		/* ML of the user upon logon */
453
454
455
456
457
458
459

	uint 	main_cmds;		/* Number of Main Commands this call */
	uint 	xfer_cmds;		/* Number of Xfer Commands this call */
	ulong	posts_read; 	/* Number of Posts read this call */
	char 	temp_uler[31];  /* User who uploaded the files to temp dir */
	char 	temp_file[41];	/* Origin of extracted temp files */
	long 	temp_cdt;		/* Credit value of file that was extracted */
460
	bool 	autohang;		/* Used for auto-hangup after transfer */
461
462
463
	size_t 	logcol; 		/* Current column of log file */
	uint 	criterrs; 		/* Critical error counter */

464
465
466
467
468
469
	uint 	curgrp; 		/* Current group */
	uint	*cursub;		/* Current sub-board for each group */
	uint	curlib; 		/* Current library */
	uint	*curdir;		/* Current directory for each library */
	uint 	*usrgrp;		/* Real group numbers */
	uint	usrgrps;		/* Number groups this user has access to */
470
	uint	usrgrp_total;	/* Total number of groups */
471
472
	uint 	*usrlib;		/* Real library numbers */
	uint	usrlibs;		/* Number of libs this user can access */
473
	uint	usrlib_total;	/* Total number of libraries */
474
475
476
477
	uint 	**usrsub;		/* Real sub numbers */
	uint	*usrsubs;		/* Num of subs with access for each grp */
	uint 	**usrdir;		/* Real dir numbers */
	uint	*usrdirs;		/* Num of dirs with access for each lib */
478
479
480
	uint	cursubnum;		/* For ARS */
	uint	curdirnum;		/* For ARS */
	ulong 	timeleft;		/* Number of seconds user has left online */
481

482
483
484
485
486
487
488
489
490
	char 	*comspec;		/* Pointer to environment variable COMSPEC */
	ushort	altul;			/* Upload to alternate path flag */
	char 	cid[LEN_CID+1]; /* Caller ID (IP Address) of current caller */
	char 	*noaccess_str;	/* Why access was denied via ARS */
	long 	noaccess_val;	/* Value of parameter not met in ARS */
	int		errorlevel; 	/* Error level of external program */

	csi_t	main_csi;		/* Main Command Shell Image */

491
492
	smbmsg_t*	current_msg;	/* For message header @-codes */

493
494
495
			/* Global command shell variables */
	uint	global_str_vars;
	char **	global_str_var;
deuce's avatar
deuce committed
496
	int32_t *	global_str_var_name;
497
	uint	global_int_vars;
deuce's avatar
deuce committed
498
	int32_t *	global_int_var;
deuce's avatar
deuce committed
499
	int32_t *	global_int_var_name;
500
501
	char *	sysvar_p[MAX_SYSVARS];
	uint	sysvar_pi;
deuce's avatar
deuce committed
502
	int32_t	sysvar_l[MAX_SYSVARS];
503
504
505
	uint	sysvar_li;

    /* ansi_term.cpp */
506
507
	const char*	ansi(int atr);			/* Returns ansi escape sequence for atr */
	char*	ansi(int atr, int curatr, char* str);
508
509
510
511
    bool	ansi_gotoxy(int x, int y);
	bool	ansi_getxy(int* x, int* y);
	bool	ansi_save(void);
	bool	ansi_restore(void);
512
513
514
515
516
517
	void	ansi_getlines(void);

			/* Command Shell Methods */
	int		exec(csi_t *csi);
	int		exec_function(csi_t *csi);
	int		exec_misc(csi_t *csi, char *path);
rswindell's avatar
rswindell committed
518
519
520
	int		exec_net(csi_t *csi);
	int		exec_msg(csi_t *csi);
	int		exec_file(csi_t *csi);
521
	long	exec_bin(const char *mod, csi_t *csi, const char* startup_dir=NULL);
522
523
	void	clearvars(csi_t *bin);
	void	freevars(csi_t *bin);
deuce's avatar
deuce committed
524
	char**	getstrvar(csi_t *bin, int32_t name);
deuce's avatar
deuce committed
525
	int32_t*	getintvar(csi_t *bin, int32_t name);
526
527
	char*	copystrvar(csi_t *csi, char *p, char *str);
	void	skipto(csi_t *csi, uchar inst);
528
	bool	ftp_cmd(csi_t* csi, SOCKET ctrl_sock, const char* cmdsrc, char* rsp);
rswindell's avatar
rswindell committed
529
	bool	ftp_put(csi_t* csi, SOCKET ctrl_sock, char* src, char* dest);
530
531
532
	bool	ftp_get(csi_t* csi, SOCKET ctrl_sock, char* src, char* dest, bool dir=false);
	SOCKET	ftp_data_sock(csi_t* csi, SOCKET ctrl_sock, SOCKADDR_IN*);

533
534
535
	bool	select_shell(void);
	bool	select_editor(void);

536
537
538
	void	sys_info(void);
	void	user_info(void);
	void	xfer_policy(void);
539
540

	void	xfer_prot_menu(enum XFER_TYPE);
541
542
543
544
	void	node_stats(uint node_num);
	void	sys_stats(void);
	void	logonlist(void);
	bool	spy(uint node_num);
545
546
547
548
549
550

	void	reset_logon_vars(void);

	uint	finduser(char *str);

	int 	sub_op(uint subnum);
deuce's avatar
deuce committed
551
	ulong	getlastmsg(uint subnum, uint32_t *ptr, time_t *t);
552
553
554
555
556
557
558
559
560
561
	time_t	getmsgtime(uint subnum, ulong ptr);
	ulong	getmsgnum(uint subnum, time_t t);

	int		dir_op(uint dirnum);
	int		getuserxfers(int fromuser, int destuser, char *fname);

	void	getmsgptrs(void);
	void	putmsgptrs(void);
	void	getusrsubs(void);
	void	getusrdirs(void);
562
563
564
	uint	getusrsub(uint subnum);
	uint	getusrgrp(uint subnum);

565
	uint	userdatdupe(uint usernumber, uint offset, uint datlen, char *dat
566
				,bool del=false, bool next=false);
567
	ulong	gettimeleft(bool handle_out_of_time=true);
568
569
570
	bool	gettimeleft_inside;

	/* str.cpp */
571
	char*	timestr(time_t intime);
572
    char	timestr_output[60];
573
	void	userlist(long mode);
574
	size_t	gettmplt(char *outstr, const char *tmplt, long mode);
575
	void	sif(char *fname, char *answers, long len);	/* Synchronet Interface File */
576
577
578
579
	void	sof(char *fname, char *answers, long len);
	void	create_sif_dat(char *siffile, char *datfile);
	void	read_sif_dat(char *siffile, char *datfile);
	void	printnodedat(uint number, node_t* node);
deuce's avatar
deuce committed
580
	bool	inputnstime32(time32_t *dt);
581
	bool	inputnstime(time_t *dt);
582
	bool	chkpass(char *pass, user_t* user, bool unique);
583
	char *	cmdstr(const char *instr, const char *fpath, const char *fspec, char *outstr);
584
585
586
587
	char	cmdstr_output[512];

	void	subinfo(uint subnum);
	void	dirinfo(uint dirnum);
588
	bool	trashcan(const char *insearch, const char *name);
589
590
	void	time_bank(void);
	void	change_user(void);
591
592
593

	/* writemsg.cpp */
	void	automsg(void);
594
	bool	writemsg(const char *str, const char *top, char *title, long mode, uint subnum
595
				,const char *dest, char** editor=NULL);
596
	char*	quotes_fname(int xedit, char* buf, size_t len);
597
	char*	msg_tmp_fname(int xedit, char* fname, size_t len);
598
	char	putmsg(const char *str, long mode);
599
	bool	msgabort(void);
600
	bool	email(int usernumber, const char *top, const char *title, long mode);
601
602
	void	forwardmail(smbmsg_t* msg, int usernum);
	void	removeline(char *str, char *str2, char num, char skip);
603
	ulong	msgeditor(char *buf, const char *top, char *title);
604
	bool	editfile(char *path, bool msg=false);
605
606
607
608
609
610
611
612
	int		loadmsg(smbmsg_t *msg, ulong number);
	ushort	chmsgattr(ushort attr);
	void	show_msgattr(ushort attr);
	void	show_msghdr(smbmsg_t* msg);
	void	show_msg(smbmsg_t* msg, long mode);
	void	msgtotxt(smbmsg_t* msg, char *str, int header, int tails);
	void	quotemsg(smbmsg_t* msg, int tails);
	void	editmsg(smbmsg_t* msg, uint subnum);
613
	void	editor_inf(int xeditnum, const char *dest, const char *title, long mode
614
615
616
				,uint subnum);
	void	copyfattach(uint to, uint from, char *title);
	bool	movemsg(smbmsg_t* msg, uint subnum);
617
618
	int		process_edited_text(char* buf, FILE* stream, long mode, unsigned* lines, unsigned maxlines);
	int		process_edited_file(const char* src, const char* dest, long mode, unsigned* lines, unsigned maxlines);
619

620
621
622
	/* postmsg.cpp */
	bool	postmsg(uint subnum, smbmsg_t* msg, long wm_mode);

623
624
625
	/* mail.cpp */
	int		delmail(uint usernumber,int which);
	void	telluser(smbmsg_t* msg);
626
	void	delallmail(uint usernumber, int which, bool permanent=true);
627
628

	/* getmsg.cpp */
629
	post_t* loadposts(uint32_t *posts, uint subnum, ulong ptr, long mode, ulong *unvalidated_num);
630
631
632
633
634

	/* readmail.cpp */
	void	readmail(uint usernumber, int sent);

	/* bulkmail.cpp */
635
636
	bool	bulkmail(uchar *ar);
	int		bulkmailhdr(smb_t*, smbmsg_t*, uint usernum);
637
638

	/* con_out.cpp */
639
640
641
642
	int		bputs(const char *str);					/* BBS puts function */
	int		rputs(const char *str, size_t len=0);	/* BBS raw puts function */
	int		bprintf(const char *fmt, ...);			/* BBS printf function */
	int		rprintf(const char *fmt, ...);			/* BBS raw printf function */
643
	void	backspace(void);				/* Output a destructive backspace via outchar */
644
	void	outchar(char ch);				/* Output a char - check echo and emu.  */
645
	void	center(char *str);
646
647
648
649
650
651
652
	void	clearline(void);
	void	cleartoeol(void);
	void	cursor_home(void);
	void	cursor_up(int count=1);
	void	cursor_down(int count=1);
	void	cursor_left(int count=1);
	void	cursor_right(int count=1);
653
	long	term_supports(long cmp_flags=0);
654
655

	/* getstr.cpp */
656
	size_t	getstr_offset;
657
	size_t	getstr(char *str, size_t length, long mode);
658
	long	getnum(ulong max, ulong dflt=0);
659
	void	insert_indicator(void);
660
661

	/* getkey.cpp */
662
	char	getkey(long mode); 		/* Waits for a key hit local or remote  */
663
	long	getkeys(const char *str, ulong max);
664
	void	ungetkey(char ch);		/* Places 'ch' into the input buffer    */
665
	char	question[MAX_TEXTDAT_ITEM_LEN+1];
666
667
	bool	yesno(const char *str);
	bool	noyes(const char *str);
668
	void	pause(void);
669
670
	const char *	mnestr;
	void	mnemonics(const char *str);
671

rswindell's avatar
rswindell committed
672
	/* inkey.cpp */
673
	char	inkey(long mode, unsigned long timeout=0);
rswindell's avatar
rswindell committed
674
675
	char	handle_ctrlkey(char ch, long mode=0);

676
677
678
	/* prntfile.cpp */
	void	printfile(char *str, long mode);
	void	printtail(char *str, int lines, long mode);
679
	void	menu(const char *code);
680

681
	int		uselect(int add, uint n, const char *title, const char *item, const uchar *ar);
682
683
684
685
686
687
688
	uint	uselect_total, uselect_num[500];

	void	redrwstr(char *strin, int i, int l, long mode);
	void	attr(int atr);				/* Change local and remote text attributes */
	void	ctrl_a(char x);			/* Peforms the Ctrl-Ax attribute changes */

	/* atcodes.cpp */
689
690
	int		show_atcode(const char *code);
	const char*	atcode(char* sp, char* str, size_t maxlen);
691
692

	/* getnode.cpp */
693
694
	int		getsmsg(int usernumber);
	int		getnmsg(void);
695
	int		whos_online(bool listself);/* Lists active nodes, returns active nodes */
696
	void	nodelist(void);
697
698
699
700
701
702
703
704
705
	int		getnodeext(uint number, char * str);
	int		getnodedat(uint number, node_t * node, bool lock);
	void	nodesync(void);
	user_t	nodesync_user;
	bool	nodesync_inside;

	/* putnode.cpp */
	int		putnodedat(uint number, node_t * node);
	int		putnodeext(uint number, char * str);
706

707
	/* login.ccp */
708
	int		login(char *str, char *pw);
709
710
711
712
713
714
	void	badlogin(char* user, char* passwd);

	/* answer.cpp */
	bool	answer();

	/* logon.ccp */
715
	bool	logon(void);
716
717

	/* logout.cpp */
718
719
720
	void	logout(void);
	void	backout(void);

721
722
723
	/* newuser.cpp */
	BOOL	newuser(void);					/* Get new user							*/

724
	/* text_sec.cpp */
725
	int		text_sec(void);						/* Text sections */
726
727

	/* readmsgs.cpp */
728
729
	int		scanposts(uint subnum, long mode, const char* find);	/* Scan sub-board */
	long	listsub(uint subnum, long mode, long start, const char* search);
730
	long	listmsgs(uint subnum, long mode, post_t* post, long start, long posts);
731
	long	searchposts(uint subnum, post_t* post, long start, long msgs, const char* find);
732
	long	showposts_toyou(post_t* post, ulong start, long posts);
733
734
735
736
	void	msghdr(smbmsg_t* msg);

	/* chat.cpp */
	void	chatsection(void);
rswindell's avatar
rswindell committed
737
	void	multinodechat(int channel=1);
738
739
	void	nodepage(void);
	void	nodemsg(void);
740
741
	uint	nodemsg_inside;
	uint	hotkey_inside;
742
743
	uchar	lastnodemsg;	/* Number of node last message was sent to */
	char	lastnodemsguser[LEN_ALIAS+1];
744
	void	guruchat(char* line, char* guru, int gurunum, char* last_answer);
745
746
	bool	guruexp(char **ptrptr, char *line);
	void	localguru(char *guru, int gurunum);
rswindell's avatar
rswindell committed
747
748
	bool	sysop_page(void);
	bool	guru_page(void);
749
750
751
752
753
754
755
756
	void	privchat(bool local=false);
	bool	chan_access(uint cnum);
	int		getnodetopage(int all, int telegram);

	/* main.cpp */
	void	printstatslog(uint node);
	ulong	logonstats(void);
	void	logoffstats(void);
757
	int		nopen(char *str, int access);
758
	int		mv(char *src, char *dest, char copy); /* fast file move/copy function */
759
	bool	chksyspass(void);
rswindell's avatar
rswindell committed
760
761
	bool	chk_ar(const uchar * str, user_t* user, client_t* client); /* checks access requirements */
	bool	ar_exp(const uchar ** ptrptr, user_t*, client_t*);
762
	void	daily_maint(void);
763
764
765
766

	/* upload.cpp */
	bool	uploadfile(file_t* f);
	char	sbbsfilename[128],sbbsfiledesc[128]; /* env vars */
767
	bool	upload(uint dirnum);
768
769
770
771
772
773
    char	upload_lastdesc[LEN_FDESC+1];
	bool	bulkupload(uint dirnum);

	/* download.cpp */
	void	downloadfile(file_t* f);
	void	notdownloaded(ulong size, time_t start, time_t end);
774
	int		protocol(prot_t* prot, enum XFER_TYPE, char *fpath, char *fspec, bool cd);
775
	const char*	protcmdline(prot_t* prot, enum XFER_TYPE type);
776
777
	void	seqwait(uint devnum);
	void	autohangup(void);
778
779
	bool	checkdszlog(file_t*);
	bool	checkprotresult(prot_t*, int error, file_t*);
780
781
	bool	sendfile(char* fname, char prot=0);
	bool	recvfile(char* fname, char prot=0);
782
783
784
785
786
787

	/* file.cpp */
	void	fileinfo(file_t* f);
	void	openfile(file_t* f);
	void	closefile(file_t* f);
	bool	removefcdt(file_t* f);
788
	bool	removefile(file_t* f);
789
790
791
792
793
794
	bool	movefile(file_t* f, int newdir);
	char *	getfilespec(char *str);
	bool	checkfname(char *fname);
	bool	addtobatdl(file_t* f);

	/* listfile.cpp */
795
796
797
	bool	listfile(const char *fname, const char *buf, uint dirnum
				,const char *search, const char letter, ulong datoffset);
	int		listfiles(uint dirnum, const char *filespec, int tofile, long mode);
798
	int		listfileinfo(uint dirnum, char *filespec, long mode);
deuce's avatar
deuce committed
799
	void	listfiletofile(char *fname, char *buf, uint dirnum, int file);
800
801
802
803
	int		batchflagprompt(uint dirnum, file_t bf[], uint total, long totalfiles);

	/* bat_xfer.cpp */
	void	batchmenu(void);
804
	void	batch_create_list(void);
805
806
	void	batch_add_list(char *list);
	bool	create_batchup_lst(void);
807
	bool	create_batchdn_lst(bool native);
808
809
810
	bool	create_bimodem_pth(void);
	void	batch_upload(void);
	void	batch_download(int xfrprot);
811
	BOOL	start_batch_download(void);
812
813
814
815
816

	/* tmp_xfer.cpp */
	void	temp_xfer(void);
	void	extract(uint dirnum);
	char *	temp_cmd(void);					/* Returns temp file command line */
817
	ulong	create_filelist(const char *name, long mode);
818
819
820
821
822
823
824
825
826
827

	/* viewfile.cpp */
	int		viewfile(file_t* f, int ext);
	void	viewfiles(uint dirnum, char *fspec);
	void	viewfilecontents(file_t* f);

	/* sortdir.cpp */
	void	resort(uint dirnum);

	/* xtrn.cpp */
828
	int		external(const char* cmdline, long mode, const char* startup_dir=NULL);
829
830
831
832
833

	/* xtrn_sec.cpp */
	int		xtrn_sec(void);					/* The external program section  */
	void	xtrndat(char* name, char* dropdir, uchar type, ulong tleft
				,ulong misc);
834
835
	bool	exec_xtrn(uint xtrnnum);			/* Executes online external program */
	bool	user_event(user_event_t);			/* Executes user event(s) */
836
837
838
	char	xtrn_access(uint xnum);			/* Does useron have access to xtrn? */
	void	moduserdat(uint xtrnnum);

rswindell's avatar
rswindell committed
839
	/* logfile.cpp */
840
	void	logentry(const char *code,const char *entry);
841
	void	log(char *str);				/* Writes 'str' to node log */
842
	void	logch(char ch, bool comma);	/* Writes 'ch' to node log */
843
844
	void	logline(const char *code,const char *str); /* Writes 'str' on it's own line in log (LOG_INFO level) */
	void	logline(int level, const char *code,const char *str);
845
	void	logofflist(void);              /* List of users logon activity */
846
	bool	syslog(const char* code, const char *entry);
847
	bool	errormsg_inside;
848
	void	errormsg(int line, const char *file, const char* action, const char *object
849
				,ulong access, const char *extinfo=NULL);
850
	BOOL	hacklog(char* prot, char* text);
851

852
853
	/* qwk.cpp */
	bool	qwklogon;
854
	ulong	qwkmail_last;
855
856
	void	qwk_sec(void);
	int		qwk_route(char *inaddr, char *fulladdr);
857
858
859
860
861
862
	uint	total_qwknodes;
	struct qwknode {
		char	id[LEN_QWKID+1];
		char	path[MAX_PATH+1];
		time_t	time;
	}* qwknode;
863
864
865
866
	void	update_qwkroute(char *via);
	void	qwk_success(ulong msgcnt, char bi, char prepack);
	void	qwksetptr(uint subnum, char *buf, int reset);
	void	qwkcfgline(char *buf,uint subnum);
867
	int		set_qwk_flag(ulong flag);
868
869
870
871
872

	/* pack_qwk.cpp */
	bool	pack_qwk(char *packet, ulong *msgcnt, bool prepack);

	/* un_qwk.cpp */
873
	bool	unpack_qwk(char *packet,uint hubnum);
874
875
876
877
878
879

	/* pack_rep.cpp */
	bool	pack_rep(uint hubnum);

	/* un_rep.cpp */
	bool	unpack_rep(char* repfile=NULL);
880
	uint	resolve_qwkconf(uint n);
881
882

	/* msgtoqwk.cpp */
883
	ulong	msgtoqwk(smbmsg_t* msg, FILE *qwk_fp, long mode, uint subnum, int conf, FILE* hdrs_dat);
884
885

	/* qwktomsg.cpp */
886
	void	qwk_new_msg(smbmsg_t* msg, char* hdrblk, long offset, str_list_t headers, bool parse_sender_hfields);
rswindell's avatar
rswindell committed
887
888
	bool	qwk_import_msg(FILE *qwk_fp, char *hdrblk, ulong blocks, char fromhub, uint subnum
				,uint touser, smbmsg_t* msg);
889
890

	/* fido.cpp */
891
	bool	netmail(const char *into, const char *subj, long mode);
892
893
894
	void	qwktonetmail(FILE *rep, char *block, char *into, uchar fromhub);
	bool	lookup_netuser(char *into);

895
896
	bool	inetmail(const char *into, const char *subj, long mode);
	bool	qnetmail(const char *into, const char *subj, long mode);
897
898

	/* useredit.cpp */
899
	void	useredit(int usernumber);
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
	int		searchup(char *search,int usernum);
	int		searchdn(char *search,int usernum);
	void	maindflts(user_t* user);
	void	purgeuser(int usernumber);

	/* ver.cpp */
	void	ver(void);

	/* scansubs.cpp */
	void	scansubs(long mode);
	void	scanallsubs(long mode);
	void	new_scan_cfg(ulong misc);
	void	new_scan_ptr_cfg(void);

	/* scandirs.cpp */
	void	scanalldirs(long mode);
	void	scandirs(long mode);

	#define nosound()
	#define checkline()

	void	catsyslog(int crash);

	/* telgate.cpp */
	void	telnet_gate(char* addr, ulong mode);	// See TG_* for mode bits

};
927

rswindell's avatar
rswindell committed
928
#endif /* __cplusplus */
929

930
931
932
#ifdef DLLEXPORT
#undef DLLEXPORT
#endif
933
934
935
#ifdef DLLCALL
#undef DLLCALL
#endif
936
#ifdef _WIN32
deuce's avatar
deuce committed
937
938
	#ifdef __MINGW32__
		#define DLLEXPORT
939
		#define DLLCALL
deuce's avatar
deuce committed
940
941
942
943
944
945
946
947
948
949
950
	#else
		#ifdef SBBS_EXPORTS
			#define DLLEXPORT	__declspec(dllexport)
		#else
			#define DLLEXPORT	__declspec(dllimport)
		#endif
		#ifdef __BORLANDC__
			#define DLLCALL __stdcall
		#else
			#define DLLCALL
		#endif
951
952
953
	#endif
#else	/* !_WIN32 */
	#define DLLEXPORT
954
	#define DLLCALL
955
956
957
958
959
960
#endif

#ifdef __cplusplus
extern "C" {
#endif

961
962
963
964
	/* main.cpp */
	DLLEXPORT int		DLLCALL sbbs_random(int);
	DLLEXPORT void		DLLCALL sbbs_srand(void);

965
	/* getstats.c */
966
	DLLEXPORT BOOL		DLLCALL getstats(scfg_t* cfg, char node, stats_t* stats);
967
968
	DLLEXPORT ulong		DLLCALL	getposts(scfg_t* cfg, uint subnum);
	DLLEXPORT long		DLLCALL getfiles(scfg_t* cfg, uint dirnum);
969

970
	/* getmail.c */
971
	DLLEXPORT int		DLLCALL getmail(scfg_t* cfg, int usernumber, BOOL sent);
972
	DLLEXPORT mail_t *	DLLCALL loadmail(smb_t* smb, uint32_t* msgs, uint usernumber
973
										,int which, long mode);
974
	DLLEXPORT void		DLLCALL freemail(mail_t* mail);
975
	DLLEXPORT void		DLLCALL delfattach(scfg_t*, smbmsg_t*);
976

977
	/* postmsg.cpp */
978
	DLLEXPORT int		DLLCALL savemsg(scfg_t*, smb_t*, smbmsg_t*, client_t*, const char* server, char* msgbuf);
979
980
	DLLEXPORT void		DLLCALL signal_sub_sem(scfg_t*, uint subnum);
	DLLEXPORT int		DLLCALL msg_client_hfields(smbmsg_t*, client_t*);
981
	DLLEXPORT char*		DLLCALL msg_program_id(char* pid);
982

983
	/* filedat.c */
984
	DLLEXPORT BOOL		DLLCALL getfileixb(scfg_t* cfg, file_t* f);
985
	DLLEXPORT BOOL		DLLCALL putfileixb(scfg_t* cfg, file_t* f);
986
987
988
989
	DLLEXPORT BOOL		DLLCALL getfiledat(scfg_t* cfg, file_t* f);
	DLLEXPORT BOOL		DLLCALL putfiledat(scfg_t* cfg, file_t* f);
	DLLEXPORT void		DLLCALL putextdesc(scfg_t* cfg, uint dirnum, ulong datoffset, char *ext);
	DLLEXPORT void		DLLCALL getextdesc(scfg_t* cfg, uint dirnum, ulong datoffset, char *ext);
990
	DLLEXPORT char*		DLLCALL getfilepath(scfg_t* cfg, file_t* f, char* path);
991
992
993
994

	DLLEXPORT BOOL		DLLCALL removefiledat(scfg_t* cfg, file_t* f);
	DLLEXPORT BOOL		DLLCALL addfiledat(scfg_t* cfg, file_t* f);
	DLLEXPORT BOOL		DLLCALL findfile(scfg_t* cfg, uint dirnum, char *filename);
995
996
	DLLEXPORT char *	DLLCALL padfname(const char *filename, char *str);
	DLLEXPORT char *	DLLCALL unpadfname(const char *filename, char *str);
997
998
	DLLEXPORT BOOL		DLLCALL rmuserxfers(scfg_t* cfg, int fromuser, int destuser, char *fname);

999
	DLLEXPORT int		DLLCALL update_uldate(scfg_t* cfg, file_t* f);
1000

1001
	/* str_util.c */
1002
1003
	DLLEXPORT char *	DLLCALL remove_ctrl_a(const char* instr, char* outstr);
	DLLEXPORT char 		DLLCALL ctrl_a_to_ascii_char(char code);
1004
	DLLEXPORT char *	DLLCALL truncstr(char* str, const char* set);
1005
	DLLEXPORT char *	DLLCALL ascii_str(uchar* str);
1006
	DLLEXPORT char		DLLCALL exascii_to_ascii_char(uchar ch);
rswindell's avatar
rswindell committed
1007
	DLLEXPORT BOOL		DLLCALL findstr(const char *insearch, const char *fname);
1008
	DLLEXPORT BOOL		DLLCALL findstr_in_string(const char* insearchof, char* string);
rswindell's avatar
rswindell committed
1009
1010
1011
1012
	DLLEXPORT BOOL		DLLCALL findstr_in_list(const char* insearchof, str_list_t list);
	DLLEXPORT BOOL		DLLCALL trashcan(scfg_t* cfg, const char *insearch, const char *name);
	DLLEXPORT char *	DLLCALL trashcan_fname(scfg_t* cfg, const char *name, char* fname, size_t);
	DLLEXPORT str_list_t DLLCALL trashcan_list(scfg_t* cfg, const char* name);
1013
	DLLEXPORT char *	DLLCALL strip_exascii(const char *str, char* dest);
1014
	DLLEXPORT char *	DLLCALL strip_space(const char *str, char* dest);
1015
1016
	DLLEXPORT char *	DLLCALL prep_file_desc(const char *str, char* dest);
	DLLEXPORT char *	DLLCALL strip_ctrl(const char *str, char* dest);
1017
	DLLEXPORT char *	DLLCALL net_addr(net_t* net);
1018
1019
	DLLEXPORT BOOL		DLLCALL valid_ctrl_a_attr(char a);
	DLLEXPORT BOOL		DLLCALL valid_ctrl_a_code(char a);
1020
	DLLEXPORT size_t	DLLCALL strip_invalid_attr(char *str);
1021
	DLLEXPORT char *	DLLCALL ultoac(ulong l,char *str);
1022
	DLLEXPORT char *	DLLCALL rot13(char* str);
1023
	DLLEXPORT uint32_t	DLLCALL str_to_bits(uint32_t currval, const char *str);
1024

1025
	/* msg_id.c */
rswindell's avatar
rswindell committed
1026
1027
	DLLEXPORT char *	DLLCALL ftn_msgid(sub_t* sub, smbmsg_t* msg, char* msgid, size_t);
	DLLEXPORT char *	DLLCALL get_msgid(scfg_t* cfg, uint subnum, smbmsg_t* msg, char* msgid, size_t);
1028

rswindell's avatar
rswindell committed
1029

1030
	/* date_str.c */
rswindell's avatar
rswindell committed
1031
	DLLEXPORT char *	DLLCALL zonestr(short zone);
1032
	DLLEXPORT time32_t	DLLCALL dstrtounix(scfg_t*, const char *str);
1033
	DLLEXPORT char *	DLLCALL unixtodstr(scfg_t*, time32_t, char *str);
1034
	DLLEXPORT char *	DLLCALL sectostr(uint sec, char *str);
1035
	DLLEXPORT char *	DLLCALL hhmmtostr(scfg_t* cfg, struct tm* tm, char* str);
1036
	DLLEXPORT char *	DLLCALL timestr(scfg_t* cfg, time32_t intime, char* str);
rswindell's avatar