Compiler quietening
[exim.git] / src / src / functions.h
CommitLineData
059ec3d9
PH
1/*************************************************
2* Exim - an Internet mail transport agent *
3*************************************************/
4
f9ba5e22 5/* Copyright (c) University of Cambridge 1995 - 2018 */
059ec3d9
PH
6/* See the file NOTICE for conditions of use and distribution. */
7
8
9/* Prototypes for functions that appear in various modules. Gathered together
10to avoid having a lot of tiddly little headers with only a couple of lines in
11them. However, some functions that are used (or not used) by utility programs
12are in in fact in separate headers. */
b66fecb4
HSHR
13#ifndef _FUNCTIONS_H_
14#define _FUNCTIONS_H_
059ec3d9 15
9f01e50d
JH
16#include <sys/time.h>
17
059ec3d9
PH
18
19#ifdef EXIM_PERL
acec9514 20extern gstring *call_perl_cat(gstring *, uschar **, uschar *,
268d00ff 21 uschar **) WARN_UNUSED_RESULT;
059ec3d9
PH
22extern void cleanup_perl(void);
23extern uschar *init_perl(uschar *);
24#endif
25
26
01603eec 27#ifndef DISABLE_TLS
a799883d
PP
28extern const char *
29 std_dh_prime_default(void);
30extern const char *
31 std_dh_prime_named(const uschar *);
9d1c15ef 32
9e4dddbd 33extern uschar * tls_cert_crl_uri(void *, uschar * mod);
9d1c15ef 34extern uschar * tls_cert_ext_by_oid(void *, uschar *, int);
9e4dddbd
JH
35extern uschar * tls_cert_issuer(void *, uschar * mod);
36extern uschar * tls_cert_not_before(void *, uschar * mod);
37extern uschar * tls_cert_not_after(void *, uschar * mod);
38extern uschar * tls_cert_ocsp_uri(void *, uschar * mod);
39extern uschar * tls_cert_serial_number(void *, uschar * mod);
40extern uschar * tls_cert_signature(void *, uschar * mod);
41extern uschar * tls_cert_signature_algorithm(void *, uschar * mod);
42extern uschar * tls_cert_subject(void *, uschar * mod);
43extern uschar * tls_cert_subject_altname(void *, uschar * mod);
44extern uschar * tls_cert_version(void *, uschar * mod);
9d1c15ef 45
59b87190 46extern uschar * tls_cert_der_b64(void * cert);
6a8a60e0
JH
47extern uschar * tls_cert_fprt_md5(void *);
48extern uschar * tls_cert_fprt_sha1(void *);
9ef9101c 49extern uschar * tls_cert_fprt_sha256(void *);
6a8a60e0 50
6b5cbf74 51extern void tls_clean_env(void);
c05bdbd6
JH
52extern BOOL tls_client_start(client_conn_ctx *, smtp_connect_args *,
53 void *, tls_support *, uschar **);
54
74f1a423 55extern void tls_close(void *, int);
925ac8e4 56extern BOOL tls_could_read(void);
b10c87b3 57extern void tls_daemon_init(void);
2944124c 58extern BOOL tls_export_cert(uschar *, size_t, void *);
059ec3d9
PH
59extern int tls_feof(void);
60extern int tls_ferror(void);
790fbb71 61extern void tls_free_cert(void **);
bd8fbe36 62extern int tls_getc(unsigned);
0d81dabc 63extern uschar *tls_getbuf(unsigned *);
584e96c6 64extern void tls_get_cache(void);
b2827658 65extern BOOL tls_import_cert(const uschar *, void **);
74f1a423 66extern int tls_read(void *, uschar *, size_t);
cf0c6164 67extern int tls_server_start(const uschar *, uschar **);
58eb016e 68extern BOOL tls_smtp_buffered(void);
059ec3d9 69extern int tls_ungetc(int);
74f1a423 70extern int tls_write(void *, const uschar *, size_t, BOOL);
3375e053 71extern uschar *tls_validate_require_cipher(void);
36f12725 72extern void tls_version_report(FILE *);
de517fd3 73# ifdef USE_OPENSSL
77bb000f 74extern BOOL tls_openssl_options_parse(uschar *, long *);
e51c7be2 75# endif
55414b25
JH
76extern uschar * tls_field_from_dn(uschar *, const uschar *);
77extern BOOL tls_is_name_for_cert(const uschar *, void *);
0e66b3b6 78
c0635b6d 79# ifdef SUPPORT_DANE
4b0fe319 80extern int tlsa_lookup(const host_item *, dns_answer *, BOOL);
0e66b3b6
JH
81# endif
82
01603eec 83#endif /*DISABLE_TLS*/
059ec3d9
PH
84
85
86/* Everything else... */
87
88extern acl_block *acl_read(uschar *(*)(void), uschar **);
89extern int acl_check(int, uschar *, uschar *, uschar **, uschar **);
05caaeaa 90extern int acl_eval(int, uschar *, uschar **, uschar **);
333eea9c 91
38a0a95f
PH
92extern tree_node *acl_var_create(uschar *);
93extern void acl_var_write(uschar *, uschar *, void *);
617d3932
JH
94
95#ifdef EXPERIMENTAL_ARC
96extern void *arc_ams_setup_sign_bodyhash(void);
97extern const uschar *arc_header_feed(gstring *, BOOL);
98extern gstring *arc_sign(const uschar *, gstring *, uschar **);
b3d9ebf5 99extern void arc_sign_init(void);
617d3932 100extern const uschar *acl_verify_arc(void);
9cffa436 101extern uschar * fn_arc_domains(void);
617d3932
JH
102#endif
103
cf0812d5 104extern void assert_no_variables(void *, int, const char *, int);
e1af7642 105extern int auth_call_pam(const uschar *, uschar **);
059ec3d9 106extern int auth_call_pwcheck(uschar *, uschar **);
93a6fce2
JH
107extern int auth_call_radius(const uschar *, uschar **);
108extern int auth_call_saslauthd(const uschar *, const uschar *,
109 const uschar *, const uschar *, uschar **);
16ff981e 110extern int auth_check_serv_cond(auth_instance *);
44bbabb5 111extern int auth_check_some_cond(auth_instance *, uschar *, uschar *, int);
37942ad8
JH
112extern int auth_client_item(void *, auth_instance *, const uschar **,
113 unsigned, int, uschar *, int);
44bbabb5 114
e1d04f48 115
1f20760b 116extern int auth_get_data(uschar **, const uschar *, int);
059ec3d9 117extern int auth_get_no64_data(uschar **, uschar *);
8238bc7b 118extern int auth_prompt(const uschar *);
37942ad8 119extern int auth_read_input(const uschar *);
adf73d37 120extern void auth_show_supported(FILE *);
059ec3d9
PH
121extern uschar *auth_xtextencode(uschar *, int);
122extern int auth_xtextdecode(uschar *, uschar **);
123
c9cf9ac4
JH
124#ifdef EXPERIMENTAL_ARC
125extern gstring *authres_arc(gstring *);
dfbcb5ac
JH
126#endif
127#ifndef DISABLE_DKIM
128extern gstring *authres_dkim(gstring *);
129#endif
1a2e76e1 130#ifdef SUPPORT_DMARC
c9cf9ac4
JH
131extern gstring *authres_dmarc(gstring *);
132#endif
133extern gstring *authres_smtpauth(gstring *);
134#ifdef SUPPORT_SPF
135extern gstring *authres_spf(gstring *);
617d3932 136#endif
dfbcb5ac 137
1f20760b 138extern uschar *b64encode(const uschar *, int);
35cf75e9 139extern int b64decode(const uschar *, uschar **);
bd8fbe36 140extern int bdat_getc(unsigned);
0d81dabc 141extern uschar *bdat_getbuf(unsigned *);
f6f4a58d 142extern int bdat_ungetc(int);
1ebe15c3
JH
143extern void bdat_flush_data(void);
144
6c6d6e48
TF
145extern void bits_clear(unsigned int *, size_t, int *);
146extern void bits_set(unsigned int *, size_t, int *);
147
57cc2785 148extern void cancel_cutthrough_connection(BOOL, const uschar *);
55414b25 149extern int check_host(void *, const uschar *, const uschar **, uschar **);
059ec3d9 150extern uschar **child_exec_exim(int, BOOL, int *, BOOL, int, ...);
55414b25
JH
151extern pid_t child_open_uid(const uschar **, const uschar **, int,
152 uid_t *, gid_t *, int *, int *, uschar *, BOOL);
bc3c7bb7 153extern BOOL cleanup_environment(void);
6851a9c5
JH
154extern void cutthrough_data_puts(uschar *, int);
155extern void cutthrough_data_put_nl(void);
e4bdf652
JH
156extern uschar *cutthrough_finaldot(void);
157extern BOOL cutthrough_flush_send(void);
158extern BOOL cutthrough_headers_send(void);
159extern BOOL cutthrough_predata(void);
57cc2785 160extern void release_cutthrough_connection(const uschar *);
059ec3d9
PH
161
162extern void daemon_go(void);
6a8f9482
TK
163
164#ifdef EXPERIMENTAL_DCC
165extern int dcc_process(uschar **);
166#endif
167
ed7f7860 168extern void debug_logging_activate(uschar *, uschar *);
b0d68adc 169extern void debug_logging_stop(void);
55414b25 170extern void debug_print_argv(const uschar **);
059ec3d9 171extern void debug_print_ids(uschar *);
e1d04f48 172extern void debug_printf_indent(const char *, ...) PRINTF_FUNCTION(1,2);
059ec3d9
PH
173extern void debug_print_string(uschar *);
174extern void debug_print_tree(tree_node *);
398f9af3 175extern void debug_vprintf(int, const char *, va_list);
6c6d6e48
TF
176extern void decode_bits(unsigned int *, size_t, int *,
177 uschar *, bit_table *, int, uschar *, int);
059ec3d9 178extern address_item *deliver_make_addr(uschar *, BOOL);
476be7e2 179extern void deliver_init(void);
817d9f57 180extern void delivery_log(int, address_item *, int, uschar *);
059ec3d9 181extern int deliver_message(uschar *, BOOL, BOOL);
e0df1c83 182extern void deliver_msglog(const char *, ...) PRINTF_FUNCTION(1,2);
059ec3d9
PH
183extern void deliver_set_expansions(address_item *);
184extern int deliver_split_address(address_item *);
185extern void deliver_succeeded(address_item *);
a39bd74d
JB
186
187extern uschar *deliver_get_sender_address (uschar *id);
57cc2785 188extern void delivery_re_exec(int);
a39bd74d 189
1ba28e2b 190extern BOOL directory_make(const uschar *, const uschar *, int, BOOL);
80a47a2c 191#ifndef DISABLE_DKIM
fc2ba7b9 192extern uschar *dkim_exim_query_dns_txt(const uschar *);
617d3932
JH
193extern void dkim_exim_sign_init(void);
194
42055a33 195extern BOOL dkim_transport_write_message(transport_ctx *,
b9df1829 196 struct ob_dkim *, const uschar ** errstr);
f7572e5a 197#endif
059ec3d9 198extern dns_address *dns_address_from_rr(dns_answer *, dns_record *);
476be7e2 199extern int dns_basic_lookup(dns_answer *, const uschar *, int);
55414b25 200extern void dns_build_reverse(const uschar *, uschar *);
a713f766 201extern time_t dns_expire_from_soa(dns_answer *, int);
8c51eead 202extern void dns_init(BOOL, BOOL, BOOL);
0539a19d 203extern BOOL dns_is_aa(const dns_answer *);
1dc92d5a 204extern BOOL dns_is_secure(const dns_answer *);
55414b25 205extern int dns_lookup(dns_answer *, const uschar *, int, const uschar **);
476be7e2 206extern void dns_pattern_init(void);
55414b25 207extern int dns_special_lookup(dns_answer *, const uschar *, int, const uschar **);
dd708fd7 208extern dns_record *dns_next_rr(const dns_answer *, dns_scan *, int);
059ec3d9 209extern uschar *dns_text_type(int);
36a3ae5f 210extern void dscp_list_to_stream(FILE *);
9e4f5962 211extern BOOL dscp_lookup(const uschar *, int, int *, int *, int *);
059ec3d9
PH
212
213extern void enq_end(uschar *);
e8ae7214 214extern BOOL enq_start(uschar *, unsigned);
0cbf2b82 215#ifndef DISABLE_EVENT
55414b25 216extern uschar *event_raise(uschar *, const uschar *, uschar *);
5ef5dd52 217extern void msg_event_raise(const uschar *, const address_item *);
774ef2d7 218#endif
b66fecb4
HSHR
219
220extern int exim_chown_failure(int, const uschar*, uid_t, gid_t);
37f3dc43 221extern const uschar * exim_errstr(int);
1d1e7973 222extern void exim_exit(int, const uschar *) NORETURN;
059ec3d9
PH
223extern void exim_nullstd(void);
224extern void exim_setugid(uid_t, gid_t, BOOL, uschar *);
f3ebb786 225extern void exim_underbar_exit(int);
059ec3d9 226extern void exim_wait_tick(struct timeval *, int);
9c695f6d
JH
227extern int exp_bool(address_item *addr,
228 uschar *mtype, uschar *mname, unsigned dgb_opt, uschar *oname, BOOL bvalue,
229 uschar *svalue, BOOL *rvalue);
059ec3d9 230extern BOOL expand_check_condition(uschar *, uschar *, uschar *);
617d3932 231extern uschar *expand_file_big_buffer(const uschar *);
93a6fce2
JH
232extern uschar *expand_string(uschar *); /* public, cannot make const */
233extern const uschar *expand_cstring(const uschar *); /* ... so use this one */
f42deca9 234extern uschar *expand_hide_passwords(uschar * );
55414b25 235extern uschar *expand_string_copy(const uschar *);
97d17305 236extern int_eximarith_t expand_string_integer(uschar *, BOOL);
d9b2312b 237extern void modify_variable(uschar *, void *);
059ec3d9 238
0a5441fc 239extern BOOL fd_ready(int, time_t);
4e71661f 240
059ec3d9
PH
241extern int filter_interpret(uschar *, int, address_item **, uschar **);
242extern BOOL filter_personal(string_item *, BOOL);
f05da2e8 243extern BOOL filter_runtest(int, uschar *, BOOL, BOOL);
059ec3d9
PH
244extern BOOL filter_system_interpret(address_item **, uschar **);
245
362145b5
JH
246extern uschar * fn_hdrs_added(void);
247
1ba28e2b 248extern void header_add(int, const char *, ...);
049782c0 249extern header_line *header_add_at_position_internal(BOOL, uschar *, BOOL, int, const char *, ...);
059ec3d9
PH
250extern int header_checkname(header_line *, BOOL);
251extern BOOL header_match(uschar *, BOOL, BOOL, string_item *, int, ...);
7cd1141b 252extern int host_address_extract_port(uschar *);
059ec3d9 253extern uschar *host_and_ident(BOOL);
55414b25
JH
254extern int host_aton(const uschar *, int *);
255extern void host_build_hostlist(host_item **, const uschar *, BOOL);
256extern ip_address_item *host_build_ifacelist(const uschar *, uschar *);
059ec3d9
PH
257extern void host_build_log_info(void);
258extern void host_build_sender_fullhost(void);
a207d5dd 259extern int host_find_byname(host_item *, const uschar *, int,
1f155f8e 260 const uschar **, BOOL);
55414b25 261extern int host_find_bydns(host_item *, const uschar *, int, uschar *, uschar *,
7cd171b7 262 uschar *, const dnssec_domains *, const uschar **, BOOL *);
059ec3d9 263extern ip_address_item *host_find_interfaces(void);
55414b25 264extern BOOL host_is_in_net(const uschar *, const uschar *, int);
059ec3d9 265extern BOOL host_is_tls_on_connect_port(int);
7cd1141b 266extern int host_item_get_port(host_item *);
059ec3d9
PH
267extern void host_mask(int, int *, int);
268extern int host_name_lookup(void);
6f0c9a4f 269extern int host_nmtoa(int, int *, int, uschar *, int);
059ec3d9
PH
270extern uschar *host_ntoa(int, const void *, uschar *, int *);
271extern int host_scan_for_local_hosts(host_item *, host_item **, BOOL *);
272
94431adb 273extern uschar *imap_utf7_encode(uschar *, const uschar *,
ed0512a1
JH
274 uschar, uschar *, uschar **);
275
83e029d5 276extern void invert_address(uschar *, uschar *);
7eb6c37c 277extern int ip_addr(void *, int, const uschar *, int);
059ec3d9 278extern int ip_bind(int, int, uschar *, int);
0ab63f3d 279extern int ip_connect(int, int, const uschar *, int, int, const blob *);
b1f8e4f8 280extern int ip_connectedsocket(int, const uschar *, int, int,
4a5cbaff 281 int, host_item *, uschar **, const blob *);
13363eba 282extern int ip_get_address_family(int);
55414b25 283extern void ip_keepalive(int, const uschar *, BOOL);
0a5441fc 284extern int ip_recv(client_conn_ctx *, uschar *, int, time_t);
059ec3d9
PH
285extern int ip_socket(int, int);
286
3e60dd41
JH
287extern int ip_tcpsocket(const uschar *, uschar **, int);
288extern int ip_unixsocket(const uschar *, uschar **);
289extern int ip_streamsocket(const uschar *, uschar **, int);
290
fc4a7f70
JH
291extern int ipv6_nmtoa(int *, uschar *);
292
921b12ca 293extern uschar *local_part_quote(uschar *);
4840604e 294extern int log_create(uschar *);
921b12ca 295extern int log_create_as_exim(uschar *);
059ec3d9
PH
296extern void log_close_all(void);
297
d185889f 298extern macro_item * macro_create(const uschar *, const uschar *, BOOL);
c246a1de
JH
299extern BOOL macro_read_assignment(uschar *);
300extern uschar *macros_expand(int, int *, BOOL *);
9cd319d9 301extern void mainlog_close(void);
8523533c 302#ifdef WITH_CONTENT_SCAN
0f0c8159 303extern int malware(const uschar *, int);
dbc4b90d 304extern int malware_in_file(uschar *);
476be7e2 305extern void malware_init(void);
c11d665d 306extern void malware_show_supported(FILE *);
8523533c 307#endif
55414b25
JH
308extern int match_address_list(const uschar *, BOOL, BOOL, const uschar **,
309 unsigned int *, int, int, const uschar **);
36d295f1 310extern int match_address_list_basic(const uschar *, const uschar **, int);
55414b25
JH
311extern int match_check_list(const uschar **, int, tree_node **, unsigned int **,
312 int(*)(void *, const uschar *, const uschar **, uschar **), void *, int,
313 const uschar *, const uschar **);
314extern int match_isinlist(const uschar *, const uschar **, int, tree_node **,
315 unsigned int *, int, BOOL, const uschar **);
316extern int match_check_string(const uschar *, const uschar *, int, BOOL, BOOL, BOOL,
317 const uschar **);
059ec3d9
PH
318extern void md5_end(md5 *, const uschar *, int, uschar *);
319extern void md5_mid(md5 *, const uschar *);
320extern void md5_start(md5 *);
321extern void millisleep(int);
8523533c
TK
322#ifdef WITH_CONTENT_SCAN
323struct mime_boundary_context;
54cdb463
PH
324extern int mime_acl_check(uschar *acl, FILE *f,
325 struct mime_boundary_context *, uschar **, uschar **);
55414b25 326extern int mime_decode(const uschar **);
f4d091fb 327extern ssize_t mime_decode_base64(FILE *, FILE *, uschar *);
55414b25 328extern int mime_regex(const uschar **);
f4d091fb 329extern void mime_set_anomaly(int);
8523533c 330#endif
059ec3d9
PH
331extern uschar *moan_check_errorcopy(uschar *);
332extern BOOL moan_skipped_syntax_errors(uschar *, error_block *, uschar *,
333 BOOL, uschar *);
e0df1c83 334extern void moan_smtp_batch(uschar *, const char *, ...) PRINTF_FUNCTION(2,3);
5455f548
JH
335extern BOOL moan_send_message(uschar *, int, error_block *eblock,
336 header_line *, FILE *, uschar *);
1ba28e2b 337extern void moan_tell_someone(uschar *, address_item *,
e0df1c83 338 const uschar *, const char *, ...) PRINTF_FUNCTION(4,5);
059ec3d9 339extern BOOL moan_to_sender(int, error_block *, header_line *, FILE *, BOOL);
0e22dfd1 340extern void moan_write_from(FILE *);
d6c829b9 341extern void moan_write_references(FILE *, uschar *);
1ba28e2b 342extern FILE *modefopen(const uschar *, const char *, mode_t);
059ec3d9 343
f9334a28 344extern int open_cutthrough_connection( address_item * addr );
e4bdf652 345
059ec3d9
PH
346extern uschar *parse_extract_address(uschar *, uschar **, int *, int *, int *,
347 BOOL);
348extern int parse_forward_list(uschar *, int, address_item **, uschar **,
55414b25 349 const uschar *, uschar *, error_block **);
059ec3d9
PH
350extern uschar *parse_find_address_end(uschar *, BOOL);
351extern uschar *parse_find_at(uschar *);
55414b25 352extern const uschar *parse_fix_phrase(const uschar *, int, uschar *, int);
30dba1e6 353extern uschar *parse_message_id(uschar *, uschar **, uschar **);
55414b25 354extern const uschar *parse_quote_2047(const uschar *, int, uschar *, uschar *, int, BOOL);
43ad7b7d 355extern uschar *parse_date_time(uschar *str, time_t *t);
17c76198 356extern int vaguely_random_number(int);
01603eec 357#ifndef DISABLE_TLS
17c76198
PP
358extern int vaguely_random_number_fallback(int);
359#endif
059ec3d9
PH
360
361extern BOOL queue_action(uschar *, int, uschar **, int, int);
362extern void queue_check_only(void);
363extern void queue_list(int, uschar **, int);
364extern void queue_count(void);
365extern void queue_run(uschar *, uschar *, BOOL);
366
367extern int random_number(int);
aeb65e91 368extern const uschar *rc_to_string(int);
e4a89c47 369extern int rda_interpret(redirect_block *, int, uschar *, uschar *,
efd9a422
MH
370 uschar *, uschar *, uschar *, ugid_block *, address_item **,
371 uschar **, error_block **, int *, uschar *);
059ec3d9
PH
372extern int rda_is_filter(const uschar *);
373extern BOOL readconf_depends(driver_instance *, uschar *);
374extern void readconf_driver_init(uschar *, driver_instance **,
375 driver_info *, int, void *, int, optionlist *, int);
376extern uschar *readconf_find_option(void *);
34e86e20 377extern void readconf_main(BOOL);
383832ef 378extern void readconf_options_from_list(optionlist *, unsigned, const uschar *, uschar *);
2be324ee 379extern BOOL readconf_print(uschar *, uschar *, BOOL);
059ec3d9
PH
380extern uschar *readconf_printtime(int);
381extern uschar *readconf_readname(uschar *, int, uschar *);
1ad6489e 382extern int readconf_readtime(const uschar *, int, BOOL);
bf3c2c6b 383extern void readconf_rest(void);
55414b25 384extern uschar *readconf_retry_error(const uschar *, const uschar *, int *, int *);
bf3c2c6b 385extern void readconf_save_config(const uschar *);
328895cc 386extern void read_message_body(BOOL);
1d1e7973 387extern void receive_bomb_out(uschar *, uschar *) NORETURN;
059ec3d9
PH
388extern BOOL receive_check_fs(int);
389extern BOOL receive_check_set_sender(uschar *);
390extern BOOL receive_msg(BOOL);
a45431fa 391extern int_eximarith_t receive_statvfs(BOOL, int *);
059ec3d9 392extern void receive_swallow_smtp(void);
8523533c 393#ifdef WITH_CONTENT_SCAN
55414b25 394extern int regex(const uschar **);
8523533c 395#endif
1dc92d5a 396extern BOOL regex_match_and_setup(const pcre *, const uschar *, int, int);
476be7e2 397extern const pcre *regex_must_compile(const uschar *, BOOL, BOOL);
059ec3d9 398extern void retry_add_item(address_item *, uschar *, int);
55414b25 399extern BOOL retry_check_address(const uschar *, host_item *, uschar *, BOOL,
059ec3d9 400 uschar **, uschar **);
55414b25
JH
401extern retry_config *retry_find_config(const uschar *, const uschar *, int, int);
402extern BOOL retry_ultimate_address_timeout(uschar *, const uschar *,
ba9af0af 403 dbdata_retry *, time_t);
059ec3d9
PH
404extern void retry_update(address_item **, address_item **, address_item **);
405extern uschar *rewrite_address(uschar *, BOOL, BOOL, rewrite_rule *, int);
406extern uschar *rewrite_address_qualify(uschar *, BOOL);
55414b25
JH
407extern header_line *rewrite_header(header_line *,
408 const uschar *, const uschar *,
059ec3d9
PH
409 rewrite_rule *, int, BOOL);
410extern uschar *rewrite_one(uschar *, int, BOOL *, BOOL, uschar *,
411 rewrite_rule *);
412extern void rewrite_test(uschar *);
413extern uschar *rfc2047_decode2(uschar *, BOOL, uschar *, int, int *, int *,
414 uschar **);
415extern int route_address(address_item *, address_item **, address_item **,
416 address_item **, address_item **, int);
55414b25
JH
417extern int route_check_prefix(const uschar *, const uschar *);
418extern int route_check_suffix(const uschar *, const uschar *);
059ec3d9 419extern BOOL route_findgroup(uschar *, gid_t *);
55414b25 420extern BOOL route_finduser(const uschar *, struct passwd **, uid_t *);
059ec3d9
PH
421extern BOOL route_find_expanded_group(uschar *, uschar *, uschar *, gid_t *,
422 uschar **);
423extern BOOL route_find_expanded_user(uschar *, uschar *, uschar *,
424 struct passwd **, uid_t *, uschar **);
425extern void route_init(void);
adf73d37 426extern void route_show_supported(FILE *);
059ec3d9
PH
427extern void route_tidyup(void);
428
55414b25 429extern uschar *search_find(void *, uschar *, uschar *, int, const uschar *, int,
059ec3d9 430 int, int *);
55414b25
JH
431extern int search_findtype(const uschar *, int);
432extern int search_findtype_partial(const uschar *, int *, const uschar **, int *,
059ec3d9
PH
433 int *);
434extern void *search_open(uschar *, int, int, uid_t *, gid_t *);
435extern void search_tidyup(void);
e0df1c83 436extern void set_process_info(const char *, ...) PRINTF_FUNCTION(1,2);
5fb822fc
JH
437extern void sha1_end(hctx *, const uschar *, int, uschar *);
438extern void sha1_mid(hctx *, const uschar *);
439extern void sha1_start(hctx *);
e4a89c47 440extern int sieve_interpret(uschar *, int, uschar *, uschar *, uschar *,
efd9a422 441 uschar *, address_item **, uschar **);
059ec3d9 442extern void sigalrm_handler(int);
58eb016e 443extern BOOL smtp_buffered(void);
059ec3d9 444extern void smtp_closedown(uschar *);
1d1e7973
JH
445extern void smtp_command_timeout_exit(void) NORETURN;
446extern void smtp_command_sigterm_exit(void) NORETURN;
447extern void smtp_data_timeout_exit(void) NORETURN;
448extern void smtp_data_sigint_exit(void) NORETURN;
a09f2942 449extern uschar *smtp_cmd_hist(void);
ee8b8090 450extern int smtp_connect(smtp_connect_args *, const blob *);
7eb6c37c 451extern int smtp_sock_connect(host_item *, int, int, uschar *,
0ab63f3d 452 transport_instance * tb, int, const blob *);
059ec3d9
PH
453extern int smtp_feof(void);
454extern int smtp_ferror(void);
455extern uschar *smtp_get_connection_info(void);
6f6dedcc 456extern BOOL smtp_get_interface(uschar *, int, address_item *,
059ec3d9
PH
457 uschar **, uschar *);
458extern BOOL smtp_get_port(uschar *, address_item *, int *, uschar *);
bd8fbe36 459extern int smtp_getc(unsigned);
0d81dabc 460extern uschar *smtp_getbuf(unsigned *);
584e96c6 461extern void smtp_get_cache(void);
059ec3d9 462extern int smtp_handle_acl_fail(int, int, uschar *, uschar *);
b4ed4da0 463extern void smtp_log_no_mail(void);
4f6ae5c3 464extern void smtp_message_code(uschar **, int *, uschar **, uschar **, BOOL);
1d1e7973 465extern void smtp_proxy_tls(void *, uschar *, size_t, int *, int) NORETURN;
251b9eb4 466extern BOOL smtp_read_response(void *, uschar *, int, int, int);
f3ebb786 467extern void *smtp_reset(void *);
a5bd321b 468extern void smtp_respond(uschar *, int, BOOL, uschar *);
8f128379 469extern void smtp_notquit_exit(uschar *, uschar *, uschar *, ...);
58c30e47 470extern void smtp_port_for_connect(host_item *, int);
059ec3d9
PH
471extern void smtp_send_prohibition_message(int, uschar *);
472extern int smtp_setup_msg(void);
473extern BOOL smtp_start_session(void);
474extern int smtp_ungetc(int);
d7b47fd0 475extern BOOL smtp_verify_helo(void);
251b9eb4 476extern int smtp_write_command(void *, int, const char *, ...) PRINTF_FUNCTION(3,4);
8523533c 477#ifdef WITH_CONTENT_SCAN
55414b25 478extern int spam(const uschar **);
040721f2 479extern FILE *spool_mbox(unsigned long *, const uschar *, uschar **);
8523533c 480#endif
4b4856ff 481extern void spool_clear_header_globals(void);
4b4856ff 482extern BOOL spool_move_message(uschar *, uschar *, uschar *, uschar *);
789f8a4f 483extern int spool_open_datafile(uschar *);
059ec3d9
PH
484extern int spool_open_temp(uschar *);
485extern int spool_read_header(uschar *, BOOL, BOOL);
486extern int spool_write_header(uschar *, int, uschar **);
bd8fbe36 487extern int stdin_getc(unsigned);
059ec3d9
PH
488extern int stdin_feof(void);
489extern int stdin_ferror(void);
490extern int stdin_ungetc(int);
f3ebb786
JH
491
492extern void store_exit(void);
acec9514
JH
493extern gstring *string_append(gstring *, int, ...) WARN_UNUSED_RESULT;
494extern gstring *string_append_listele(gstring *, uschar, const uschar *) WARN_UNUSED_RESULT;
495extern gstring *string_append_listele_n(gstring *, uschar, const uschar *, unsigned) WARN_UNUSED_RESULT;
bce15b62 496extern gstring *string_append2_listele_n(gstring *, const uschar *, const uschar *, unsigned) WARN_UNUSED_RESULT;
059ec3d9 497extern uschar *string_base62(unsigned long int);
acec9514
JH
498extern gstring *string_cat (gstring *, const uschar * ) WARN_UNUSED_RESULT;
499extern gstring *string_catn(gstring *, const uschar *, int) WARN_UNUSED_RESULT;
84bbb4d8 500extern int string_compare_by_pointer(const void *, const void *);
059ec3d9 501extern uschar *string_copy_dnsdomain(uschar *);
55414b25 502extern uschar *string_copy_malloc(const uschar *);
55414b25 503extern uschar *string_dequote(const uschar **);
059ec3d9 504extern uschar *string_format_size(int, uschar *);
55414b25 505extern int string_interpret_escape(const uschar **);
b1f8e4f8 506extern int string_is_ip_address(const uschar *, int *);
8c5d388a 507#ifdef SUPPORT_I18N
0d7911ea
JH
508extern BOOL string_is_utf8(const uschar *);
509#endif
55414b25 510extern uschar *string_nextinlist(const uschar **, int *, uschar *, int);
55414b25 511extern const uschar *string_printing2(const uschar *, BOOL);
e28326d8 512extern uschar *string_split_message(uschar *);
c7396ac5 513extern uschar *string_unprinting(uschar *);
8c5d388a 514#ifdef SUPPORT_I18N
3c8b3577 515extern uschar *string_address_utf8_to_alabel(const uschar *, uschar **);
0d7911ea
JH
516extern uschar *string_domain_alabel_to_utf8(const uschar *, uschar **);
517extern uschar *string_domain_utf8_to_alabel(const uschar *, uschar **);
518extern uschar *string_localpart_alabel_to_utf8(const uschar *, uschar **);
519extern uschar *string_localpart_utf8_to_alabel(const uschar *, uschar **);
520#endif
f3ebb786
JH
521
522#define string_format(buf, siz, fmt, ...) \
c4efe382 523 string_format_trc(buf, siz, US __FUNCTION__, __LINE__, fmt, __VA_ARGS__)
f3ebb786
JH
524extern BOOL string_format_trc(uschar *, int, const uschar *, unsigned,
525 const char *, ...) ALMOST_PRINTF(5,6);
526
527#define string_vformat(g, flgs, fmt, ap) \
528 string_vformat_trc(g, US __FUNCTION__, __LINE__, \
529 STRING_SPRINTF_BUFFER_SIZE, flgs, fmt, ap)
530extern gstring *string_vformat_trc(gstring *, const uschar *, unsigned,
531 unsigned, unsigned, const char *, va_list);
532
533#define string_open_failed(eno, fmt, ...) \
c4efe382 534 string_open_failed_trc(eno, US __FUNCTION__, __LINE__, fmt, __VA_ARGS__)
f3ebb786
JH
535extern uschar *string_open_failed_trc(int, const uschar *, unsigned,
536 const char *, ...) PRINTF_FUNCTION(4,5);
537
1ba28e2b
PP
538extern int strcmpic(const uschar *, const uschar *);
539extern int strncmpic(const uschar *, const uschar *, int);
059ec3d9
PH
540extern uschar *strstric(uschar *, uschar *, BOOL);
541
44416341 542extern int test_harness_fudged_queue_time(int);
10ac8d7f
JH
543#ifdef EXIM_TFO_PROBE
544extern void tfo_probe(void);
545#endif
32dfdf8b 546extern void tls_modify_variables(tls_support *);
059ec3d9 547extern uschar *tod_stamp(int);
41afb5cb 548
55414b25 549extern BOOL transport_check_waiting(const uschar *, const uschar *, int, uschar *,
a39bd74d 550 BOOL *, oicf, void*);
059ec3d9 551extern void transport_init(void);
57cc2785
JH
552extern void transport_do_pass_socket(const uschar *, const uschar *,
553 const uschar *, uschar *, int);
55414b25 554extern BOOL transport_pass_socket(const uschar *, const uschar *, const uschar *, uschar *,
059ec3d9
PH
555 int);
556extern uschar *transport_rcpt_address(address_item *, BOOL);
55414b25
JH
557extern BOOL transport_set_up_command(const uschar ***, uschar *,
558 BOOL, int, address_item *, uschar *, uschar **);
059ec3d9 559extern void transport_update_waiting(host_item *, uschar *);
42055a33
JH
560extern BOOL transport_write_block(transport_ctx *, uschar *, int, BOOL);
561extern void transport_write_reset(int);
1ba28e2b 562extern BOOL transport_write_string(int, const char *, ...);
42055a33
JH
563extern BOOL transport_headers_send(transport_ctx *,
564 BOOL (*)(transport_ctx *, uschar *, int));
adf73d37 565extern void transport_show_supported(FILE *);
42055a33 566extern BOOL transport_write_message(transport_ctx *, int);
059ec3d9
PH
567extern void tree_add_duplicate(uschar *, address_item *);
568extern void tree_add_nonrecipient(uschar *);
569extern void tree_add_unusable(host_item *);
b4f579d1 570extern void tree_dup(tree_node **, tree_node *);
059ec3d9 571extern int tree_insertnode(tree_node **, tree_node *);
55414b25 572extern tree_node *tree_search(tree_node *, const uschar *);
059ec3d9 573extern void tree_write(tree_node *, FILE *);
38a0a95f 574extern void tree_walk(tree_node *, void (*)(uschar*, uschar*, void*), void *);
059ec3d9 575
8523533c
TK
576#ifdef WITH_CONTENT_SCAN
577extern void unspool_mbox(void);
578#endif
8c5d388a 579#ifdef SUPPORT_I18N
fc362fc5
JH
580extern void utf8_version_report(FILE *);
581#endif
8523533c 582
4deaf07d
PH
583extern int verify_address(address_item *, FILE *, int, int, int, int,
584 uschar *, uschar *, BOOL *);
379ba7d0 585extern int verify_check_dnsbl(int, const uschar **, uschar **);
4deaf07d 586extern int verify_check_header_address(uschar **, uschar **, int, int, int,
fe5b5d0b 587 uschar *, uschar *, int, int *);
059ec3d9 588extern int verify_check_headers(uschar **);
770747fd 589extern int verify_check_header_names_ascii(uschar **);
059ec3d9 590extern int verify_check_host(uschar **);
7c498df1 591extern int verify_check_notblind(BOOL);
3fb3231c 592extern int verify_check_given_host(const uschar **, const host_item *);
55414b25
JH
593extern int verify_check_this_host(const uschar **, unsigned int *,
594 const uschar*, const uschar *, const uschar **);
059ec3d9
PH
595extern address_item *verify_checked_sender(uschar *);
596extern void verify_get_ident(int);
597extern BOOL verify_sender(int *, uschar **);
598extern BOOL verify_sender_preliminary(int *, uschar **);
599extern void version_init(void);
600
42055a33 601extern BOOL write_chunk(transport_ctx *, uschar *, int);
17c76198
PP
602extern ssize_t write_to_fd_buf(int, const uschar *, size_t);
603
6940b3df 604
7172970e 605#if !defined(MACRO_PREDEF) && !defined(COMPILE_UTILITY)
b66fecb4 606/* exim_chown - in some NFSv4 setups *seemes* to be an issue with
6940b3df 607chown(<exim-uid>, <exim-gid>).
b66fecb4 608
6940b3df
JH
609Probably because the idmapping is broken, misconfigured or set up in
610an unusal way. (see Bug 2931). As I'm not sure, if this was a single
611case of misconfiguration, or if there are more such broken systems
612out, I try to impose as least impact as possible and for now just write
613a panic log entry pointing to the bug report. You're encouraged to
614contact the developers, if you experience this issue.
b66fecb4 615
6940b3df
JH
616fd the file descriptor (or -1 if not valid)
617name the file name for error messages or for file operations,
618 if fd is < 0
619owner the owner
620group the group
b66fecb4 621
6940b3df 622returns 0 on success, -1 on failure */
b66fecb4 623
6940b3df 624static inline int
b66fecb4
HSHR
625exim_fchown(int fd, uid_t owner, gid_t group, const uschar *name)
626{
6940b3df
JH
627return fchown(fd, owner, group)
628 ? exim_chown_failure(fd, name, owner, group) : 0;
b66fecb4
HSHR
629}
630
6940b3df 631static inline int
b66fecb4
HSHR
632exim_chown(const uschar *name, uid_t owner, gid_t group)
633{
6940b3df
JH
634return chown(CCS name, owner, group)
635 ? exim_chown_failure(-1, name, owner, group) : 0;
b66fecb4 636}
e59797e3
JH
637
638#endif /* !MACRO_PREDEF && !COMPILE_UTILITY */
639/******************************************************************************/
640/* String functions */
641
f3ebb786 642#if !defined(MACRO_PREDEF)
e59797e3
JH
643/*************************************************
644* Copy and save string *
645*************************************************/
646
647/* This function assumes that memcpy() is faster than strcpy().
648*/
649
e59797e3 650static inline uschar *
f3ebb786 651string_copy_taint_trc(const uschar *s, BOOL tainted, const char * func, int line)
e59797e3
JH
652{
653int len = Ustrlen(s) + 1;
f3ebb786 654uschar *ss = store_get_3(len, tainted, func, line);
e59797e3
JH
655memcpy(ss, s, len);
656return ss;
657}
658
f3ebb786
JH
659#define string_copy_taint(s, tainted) \
660 string_copy_taint_trc((s), tainted, __FUNCTION__, __LINE__)
661
662static inline uschar *
663string_copy(const uschar * s)
664{
665return string_copy_taint((s), is_tainted(s));
666}
667
e59797e3
JH
668
669/*************************************************
670* Copy, lowercase and save string *
671*************************************************/
672
673/*
674Argument: string to copy
675Returns: copy of string in new store, with letters lowercased
676*/
677
678static inline uschar *
679string_copylc(const uschar *s)
680{
f3ebb786 681uschar *ss = store_get(Ustrlen(s) + 1, is_tainted(s));
e59797e3
JH
682uschar *p = ss;
683while (*s != 0) *p++ = tolower(*s++);
684*p = 0;
685return ss;
686}
687
688
689
690/*************************************************
691* Copy and save string, given length *
692*************************************************/
693
694/* It is assumed the data contains no zeros. A zero is added
695onto the end.
696
697Arguments:
698 s string to copy
699 n number of characters
700
701Returns: copy of string in new store
702
703This is an API for local_scan hence not static.
704*/
705
706static inline uschar *
707string_copyn(const uschar *s, int n)
708{
f3ebb786 709uschar *ss = store_get(n + 1, is_tainted(s));
e59797e3
JH
710Ustrncpy(ss, s, n);
711ss[n] = 0;
712return ss;
713}
714
715/*************************************************
716* Copy, lowercase, and save string, given length *
717*************************************************/
718
719/* It is assumed the data contains no zeros. A zero is added
720onto the end.
721
722Arguments:
723 s string to copy
724 n number of characters
725
726Returns: copy of string in new store, with letters lowercased
727*/
728
729static inline uschar *
730string_copynlc(uschar *s, int n)
731{
f3ebb786 732uschar *ss = store_get(n + 1, is_tainted(s));
e59797e3
JH
733uschar *p = ss;
734while (n-- > 0) *p++ = tolower(*s++);
735*p = 0;
736return ss;
737}
738
739
1a44d9d7 740# ifndef COMPILE_UTILITY
f3ebb786
JH
741/*************************************************
742* Copy and save string in longterm store *
743*************************************************/
744
745/* This function assumes that memcpy() is faster than strcpy().
746
747Argument: string to copy
748Returns: copy of string in new store
749*/
750
751static inline uschar *
752string_copy_perm(const uschar *s, BOOL force_taint)
753{
754int old_pool = store_pool;
755int len = Ustrlen(s) + 1;
756uschar *ss;
757
758store_pool = POOL_PERM;
759ss = store_get(len, force_taint || is_tainted(s));
760memcpy(ss, s, len);
761store_pool = old_pool;
762return ss;
763}
1a44d9d7 764# endif
f3ebb786
JH
765
766
767
768/* sprintf into a buffer, taint-unchecked */
769
770static inline void
771string_format_nt(uschar * buf, int siz, const char * fmt, ...)
772{
773gstring gs = { .size = siz, .ptr = 0, .s = buf };
774va_list ap;
775va_start(ap, fmt);
776(void) string_vformat(&gs, SVFMT_TAINT_NOCHK, fmt, ap);
777va_end(ap);
778}
779
780
781
e59797e3
JH
782/******************************************************************************/
783/* Growable-string functions */
784
f3ebb786
JH
785/* Create a growable-string with some preassigned space */
786
787#define string_get_tainted(size, tainted) \
788 string_get_tainted_trc((size), (tainted), __FUNCTION__, __LINE__)
e59797e3
JH
789
790static inline gstring *
f3ebb786 791string_get_tainted_trc(unsigned size, BOOL tainted, const char * func, unsigned line)
e59797e3 792{
f3ebb786 793gstring * g = store_get_3(sizeof(gstring) + size, tainted, func, line);
e59797e3
JH
794g->size = size;
795g->ptr = 0;
796g->s = US(g + 1);
797return g;
798}
799
f3ebb786
JH
800#define string_get(size) \
801 string_get_trc((size), __FUNCTION__, __LINE__)
802
803static inline gstring *
804string_get_trc(unsigned size, const char * func, unsigned line)
805{
806return string_get_tainted_trc(size, FALSE, func, line);
807}
808
e59797e3
JH
809/* NUL-terminate the C string in the growable-string, and return it. */
810
811static inline uschar *
812string_from_gstring(gstring * g)
813{
814if (!g) return NULL;
815g->s[g->ptr] = '\0';
816return g->s;
817}
818
f3ebb786
JH
819
820#define gstring_release_unused(g) \
821 gstring_release_unused_trc(g, __FUNCTION__, __LINE__)
822
e59797e3 823static inline void
f3ebb786 824gstring_release_unused_trc(gstring * g, const char * file, unsigned line)
e59797e3 825{
f3ebb786
JH
826if (g) store_release_above_3(g->s + (g->size = g->ptr + 1), file, line);
827}
828
829
830/* sprintf-append to a growable-string */
831
832#define string_fmt_append(g, fmt, ...) \
833 string_fmt_append_f_trc(g, US __FUNCTION__, __LINE__, \
c4efe382 834 SVFMT_EXTEND|SVFMT_REBUFFER, fmt, __VA_ARGS__)
f3ebb786
JH
835
836#define string_fmt_append_f(g, flgs, fmt, ...) \
837 string_fmt_append_f_trc(g, US __FUNCTION__, __LINE__, \
c4efe382 838 flgs, fmt, __VA_ARGS__)
f3ebb786
JH
839
840static inline gstring *
841string_fmt_append_f_trc(gstring * g, const uschar * func, unsigned line,
842 unsigned flags, const char *format, ...)
843{
844va_list ap;
845va_start(ap, format);
846g = string_vformat_trc(g, func, line, STRING_SPRINTF_BUFFER_SIZE,
847 flags, format, ap);
848va_end(ap);
849return g;
e59797e3
JH
850}
851
852/******************************************************************************/
8743d3ac
JH
853
854#define store_get_dns_answer() store_get_dns_answer_trc(CUS __FUNCTION__, __LINE__)
855
856static inline dns_answer *
857store_get_dns_answer_trc(const uschar * func, unsigned line)
858{
859return store_get_3(sizeof(dns_answer), TRUE, CCS func, line); /* use tainted mem */
860}
861
59a93276
JH
862/******************************************************************************/
863/* Routines with knowledge of spool layout */
864
865# ifndef COMPILE_UTILITY
866static inline void
867spool_pname_buf(uschar * buf, int len)
868{
869snprintf(CS buf, len, "%s/%s/input", spool_directory, queue_name);
870}
871
872static inline uschar *
873spool_dname(const uschar * purpose, uschar * subdir)
874{
875return string_sprintf("%s/%s/%s/%s",
876 spool_directory, queue_name, purpose, subdir);
877}
878# endif
879
880static inline uschar *
fc7bae7f 881spool_q_sname(const uschar * purpose, const uschar * q, uschar * subdir)
59a93276
JH
882{
883return string_sprintf("%s%s%s%s%s",
fc7bae7f 884 q, *q ? "/" : "",
59a93276
JH
885 purpose,
886 *subdir ? "/" : "", subdir);
887}
888
fc7bae7f
JH
889static inline uschar *
890spool_sname(const uschar * purpose, uschar * subdir)
891{
892return spool_q_sname(purpose, queue_name, subdir);
893}
894
895static inline uschar *
896spool_q_fname(const uschar * purpose, const uschar * q,
897 const uschar * subdir, const uschar * fname, const uschar * suffix)
898{
899return string_sprintf("%s/%s/%s/%s/%s%s",
900 spool_directory, q, purpose, subdir, fname, suffix);
901}
902
59a93276
JH
903static inline uschar *
904spool_fname(const uschar * purpose, const uschar * subdir, const uschar * fname,
905 const uschar * suffix)
906{
fc7bae7f 907return spool_q_fname(purpose, queue_name, subdir, fname, suffix);
59a93276
JH
908}
909
9b62f401 910static inline void
59a93276
JH
911set_subdir_str(uschar * subdir_str, const uschar * name,
912 int search_sequence)
913{
914subdir_str[0] = split_spool_directory == (search_sequence == 0)
915 ? name[5] : '\0';
916subdir_str[1] = '\0';
917}
918
919/******************************************************************************/
9f01e50d
JH
920static inline void
921timesince(struct timeval * diff, struct timeval * then)
922{
923gettimeofday(diff, NULL);
924diff->tv_sec -= then->tv_sec;
925if ((diff->tv_usec -= then->tv_usec) < 0)
926 {
927 diff->tv_sec--;
928 diff->tv_usec += 1000*1000;
929 }
930}
931
932static inline uschar *
933string_timediff(struct timeval * diff)
934{
935static uschar buf[sizeof("0.000s")];
936
937if (diff->tv_sec >= 5 || !LOGGING(millisec))
938 return readconf_printtime((int)diff->tv_sec);
939
b09c5f34 940snprintf(CS buf, sizeof(buf), "%u.%03us", (uint)diff->tv_sec, (uint)diff->tv_usec/1000);
9f01e50d
JH
941return buf;
942}
943
944
945static inline uschar *
946string_timesince(struct timeval * then)
947{
948struct timeval diff;
949timesince(&diff, then);
950return string_timediff(&diff);
951}
952
953static inline void
954report_time_since(struct timeval * t0, uschar * where)
955{
956# ifdef MEASURE_TIMING
957struct timeval diff;
958timesince(&diff, t0);
959fprintf(stderr, "%d %s:\t%ld.%06ld\n",
960 (uint)getpid(), where, (long)diff.tv_sec, (long)diff.tv_usec);
961# endif
962}
963
964
965static inline void
966testharness_pause_ms(int millisec)
967{
968#ifndef MEASURE_TIMING
969if (f.running_in_test_harness) millisleep(millisec);
970#endif
971}
972
eb2fb50d 973#endif /* !MACRO_PREDEF */
b66fecb4
HSHR
974
975#endif /* _FUNCTIONS_H_ */
acec9514 976
9d1c15ef
JH
977/* vi: aw
978*/
059ec3d9 979/* End of functions.h */