/* * misc.c, SJ */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "misc.h" #include "smtpcodes.h" #include "errmsg.h" #include "config.h" #include "tai.h" int get_build(){ return BUILD; } void __fatal(char *s){ fprintf(stderr, "%s\n", s); exit(1); } /* * calculate the difference betwwen two timevals in [usec] */ long tvdiff(struct timeval a, struct timeval b){ double res; res = (a.tv_sec * 1000000 + a.tv_usec) - (b.tv_sec * 1000000 + b.tv_usec); return (long) res; } /* * search something in a buffer */ int searchStringInBuffer(char *s, int len1, char *what, int len2){ int i, k, r; for(i=0; i 0){ p[k] = to; k++; } } else { p[k] = p[i]; k++; } } p[k] = '\0'; } /* * split a string by a character as delimiter */ char *split(char *row, int ch, char *s, int size){ char *r; if(row == NULL || s == NULL) return NULL; r = strchr(row, ch); if(r) *r = '\0'; snprintf(s, size, "%s", row); if(r){ *r = ch; r++; } return r; } /* * split a string by a string as delimiter */ char *split_str(char *row, char *what, char *s, int size){ char *r; int len; memset(s, 0, size); if(row == NULL) return NULL; r = strstr(row, what); if(r == NULL){ len = strlen(row); if(len > size) len = size; } else { len = strlen(row) - strlen(r); if(len > size) len = size; r += strlen(what); } if(s != NULL){ strncpy(s, row, len); s[len] = '\0'; } return r; } /* * trim trailing CR-LF */ int trimBuffer(char *s){ int n=0; char *p; p = strrchr(s, '\n'); if(p){ *p = '\0'; n++; } p = strrchr(s, '\r'); if(p){ *p = '\0'; n++; } return n; } int extractEmail(char *rawmail, char *email){ char *p; memset(email, 0, SMALLBUFSIZE); p = strchr(rawmail, '<'); if(p){ snprintf(email, SMALLBUFSIZE-1, "%s", p+1); p = strchr(email, '>'); if(p){ *p = '\0'; return 1; } } return 0; } void create_id(char *id){ int i; unsigned char buf[RND_STR_LEN/2]; memset(id, 0, SMALLBUFSIZE); get_random_bytes(buf, RND_STR_LEN/2); for(i=0; i < RND_STR_LEN/2; i++){ sprintf(id, "%02x", buf[i]); id += 2; } } /* * reading from pool */ int get_random_bytes(unsigned char *buf, int len){ int fd, ret=0; struct taia now; char nowpack[TAIA_PACK]; /* the first 12 bytes are the taia timestamp */ taia_now(&now); taia_pack(nowpack, &now); memcpy(buf, nowpack, 12); fd = open(RANDOM_POOL, O_RDONLY); if(fd == -1) return ret; if(readFromEntropyPool(fd, buf+12, len-12) != len-12){ syslog(LOG_PRIORITY, "%s: %s", ERR_CANNOT_READ_FROM_POOL, RANDOM_POOL); } close(fd); return ret; } /* * read random data from entropy pool */ int readFromEntropyPool(int fd, void *_s, size_t n){ char *s = _s; ssize_t res, pos = 0; while(n > pos){ res = read(fd, s + pos, n - pos); switch(res){ case -1: continue; case 0: return res; default : pos += res; } } return pos; } /* * recv() with timeout */ int recvtimeout(int s, char *buf, int len, int timeout){ fd_set fds; int n; struct timeval tv; memset(buf, 0, len); FD_ZERO(&fds); FD_SET(s, &fds); tv.tv_sec = timeout; tv.tv_usec = TIMEOUT_USEC; n = select(s+1, &fds, NULL, NULL, &tv); if (n == 0) return -2; // timeout! if (n == -1) return -1; // error return recv(s, buf, len, 0); } int write1(int sd, char *buf, int use_ssl, SSL *ssl){ int n; if(use_ssl == 1) n = SSL_write(ssl, buf, strlen(buf)); else n = send(sd, buf, strlen(buf), 0); return n; } int ssl_want_retry(SSL *ssl, int ret, int timeout){ int i; fd_set fds; struct timeval tv; int sock; // something went wrong. I'll retry, die quietly, or complain i = SSL_get_error(ssl, ret); if(i == SSL_ERROR_NONE) return 1; tv.tv_sec = timeout/1000; tv.tv_usec = 0; FD_ZERO(&fds); switch(i){ case SSL_ERROR_WANT_READ: // pause until the socket is readable sock = SSL_get_rfd(ssl); FD_SET(sock, &fds); i = select(sock+1, &fds, 0, 0, &tv); break; case SSL_ERROR_WANT_WRITE: // pause until the socket is writeable sock = SSL_get_wfd(ssl); FD_SET(sock, &fds); i = select(sock+1, 0, &fds, 0, &tv); break; case SSL_ERROR_ZERO_RETURN: // the sock closed, just return quietly i = 0; break; default: // ERROR - unexpected error code i = -1; break; }; return i; } int ssl_read_timeout(SSL *ssl, void *buf, int len, int timeout){ int i; while(1){ i = SSL_read(ssl, (char*)buf, len); if(i > 0) break; i = ssl_want_retry(ssl, i, timeout); if(i <= 0) break; } return i; } int recvtimeoutssl(int s, char *buf, int len, int timeout, int use_ssl, SSL *ssl){ memset(buf, 0, len); if(use_ssl == 1){ return ssl_read_timeout(ssl, buf, len-1, timeout); } else { return recvtimeout(s, buf, len-1, timeout); } } /* * is it a valid dotted IPv4 address */ int isDottedIPv4Address(char *s){ struct in_addr addr; if(inet_aton(s, &addr) == 0) return 0; return 1; } void write_pid_file(char *pidfile){ FILE *f; f = fopen(pidfile, "w"); if(f){ fprintf(f, "%d", (int)getpid()); fclose(f); } else syslog(LOG_PRIORITY, "cannot write pidfile: %s", pidfile); } int drop_privileges(struct passwd *pwd){ if(pwd->pw_uid > 0 && pwd->pw_gid > 0){ if(getgid() != pwd->pw_gid){ if(setgid(pwd->pw_gid)) return -1; } if(getuid() != pwd->pw_uid){ if(setuid(pwd->pw_uid)) return -1; } } return 0; } int is_email_address_on_my_domains(char *email, struct __config *cfg){ int rc=0; char *p, *q=NULL; if(email == NULL || cfg->mydomains == NULL) return rc; p = strchr(email, '@'); if(!p) return rc; if(strlen(p) < 3) return rc; q = strrchr(p+1, ' '); if(q) *q = '\0'; if(strcasestr(cfg->mydomains, p+1)) rc = 1; if(q) *q = ' '; return rc; } void init_session_data(struct session_data *sdata){ int i; sdata->fd = -1; create_id(&(sdata->ttmpfile[0])); unlink(sdata->ttmpfile); snprintf(sdata->filename, SMALLBUFSIZE-1, "%s", sdata->ttmpfile); snprintf(sdata->tmpframe, SMALLBUFSIZE-1, "%s.m", sdata->ttmpfile); unlink(sdata->tmpframe); memset(sdata->mailfrom, 0, SMALLBUFSIZE); snprintf(sdata->client_addr, SMALLBUFSIZE-1, "null"); memset(sdata->attachments, 0, SMALLBUFSIZE); memset(sdata->fromemail, 0, SMALLBUFSIZE); sdata->restored_copy = 0; sdata->internal_sender = sdata->internal_recipient = sdata->external_recipient = 0; sdata->direction = 0; sdata->hdr_len = 0; sdata->tot_len = 0; sdata->num_of_rcpt_to = 0; sdata->ms_journal = 0; sdata->ms_journal_dropped = 0; sdata->journal_envelope_length = 0; sdata->journal_bottom_length = 0; sdata->tre = '-'; sdata->rav = AVIR_OK; sdata->spam_message = 0; sdata->__acquire = sdata->__parsed = sdata->__av = sdata->__store = sdata->__compress = sdata->__encrypt = 0; for(i=0; ircptto[i], 0, SMALLBUFSIZE); time(&(sdata->now)); sdata->sent = sdata->delivered = sdata->retained = sdata->now; } int read_from_stdin(struct session_data *sdata){ int fd, rc=ERR, n; char buf[MAXBUFSIZE]; fd = open(sdata->ttmpfile, O_CREAT|O_EXCL|O_RDWR|O_TRUNC, S_IRUSR|S_IWUSR); if(fd == -1){ syslog(LOG_PRIORITY, "%s: cannot open ttmpfile", sdata->ttmpfile); return rc; } while((n = read(0, buf, sizeof(buf))) > 0){ sdata->tot_len += write(fd, buf, n); } if(fsync(fd)){ syslog(LOG_PRIORITY, "%s: fsync() error", sdata->ttmpfile); } else rc = OK; close(fd); return rc; } void strtolower(char *s){ for(; *s; s++) *s = tolower(*s); } unsigned long resolve_host(char *host){ struct hostent *h; struct in_addr addr; if(!host) return 0; if((addr.s_addr = inet_addr(host)) == -1){ if((h = gethostbyname(host)) == NULL){ return 0; } else return *(unsigned long*)h->h_addr; } else return addr.s_addr; } #ifndef _GNU_SOURCE char *strcasestr(const char *s, const char *find){ char c, sc; size_t len; if((c = *find++) != 0){ c = tolower((unsigned char)c); len = strlen(find); do { do { if((sc = *s++) == 0) return (NULL); } while((char)tolower((unsigned char)sc) != c); } while (strncasecmp(s, find, len) != 0); s--; } return((char*)s); } #endif