123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413 |
- /*
- * mbsync - mailbox synchronizer
- * Copyright (C) 2000-2002 Michael R. Elkins <me@mutt.org>
- * Copyright (C) 2002-2006 Oswald Buddenhagen <ossi@users.sf.net>
- *
- * 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.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software Foundation,
- * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
- *
- * As a special exception, mbsync may be linked with the OpenSSL library,
- * despite that library's more restrictive license.
- */
- #include "isync.h"
- #include <stdlib.h>
- #include <unistd.h>
- #include <fcntl.h>
- #include <string.h>
- #include <pwd.h>
- #include <ctype.h>
- int DFlags, Ontty;
- static int need_nl;
- void
- debug( const char *msg, ... )
- {
- va_list va;
- if (DFlags & DEBUG) {
- va_start( va, msg );
- vprintf( msg, va );
- va_end( va );
- fflush( stdout );
- need_nl = 0;
- }
- }
- void
- debugn( const char *msg, ... )
- {
- va_list va;
- if (DFlags & DEBUG) {
- va_start( va, msg );
- vprintf( msg, va );
- va_end( va );
- fflush( stdout );
- need_nl = Ontty;
- }
- }
- void
- info( const char *msg, ... )
- {
- va_list va;
- if (!(DFlags & QUIET)) {
- va_start( va, msg );
- vprintf( msg, va );
- va_end( va );
- fflush( stdout );
- need_nl = 0;
- }
- }
- void
- infon( const char *msg, ... )
- {
- va_list va;
- if (!(DFlags & QUIET)) {
- va_start( va, msg );
- vprintf( msg, va );
- va_end( va );
- fflush( stdout );
- need_nl = Ontty;
- }
- }
- void
- warn( const char *msg, ... )
- {
- va_list va;
- if (!(DFlags & VERYQUIET)) {
- if (need_nl) {
- putchar( '\n' );
- need_nl = 0;
- }
- va_start( va, msg );
- vfprintf( stderr, msg, va );
- va_end( va );
- }
- }
- void
- error( const char *msg, ... )
- {
- va_list va;
- if (need_nl) {
- putchar( '\n' );
- need_nl = 0;
- }
- va_start( va, msg );
- vfprintf( stderr, msg, va );
- va_end( va );
- }
- char *
- next_arg( char **s )
- {
- char *ret;
- if (!s || !*s)
- return 0;
- while (isspace( (unsigned char) **s ))
- (*s)++;
- if (!**s) {
- *s = 0;
- return 0;
- }
- if (**s == '"') {
- ++*s;
- ret = *s;
- *s = strchr( *s, '"' );
- } else {
- ret = *s;
- while (**s && !isspace( (unsigned char) **s ))
- (*s)++;
- }
- if (*s) {
- if (**s)
- *(*s)++ = 0;
- if (!**s)
- *s = 0;
- }
- return ret;
- }
- void
- add_string_list( string_list_t **list, const char *str )
- {
- string_list_t *elem;
- int len;
- len = strlen( str );
- elem = nfmalloc( sizeof(*elem) + len );
- elem->next = *list;
- *list = elem;
- memcpy( elem->string, str, len + 1 );
- }
- void
- free_string_list( string_list_t *list )
- {
- string_list_t *tlist;
- for (; list; list = tlist) {
- tlist = list->next;
- free( list );
- }
- }
- void
- free_generic_messages( message_t *msgs )
- {
- message_t *tmsg;
- for (; msgs; msgs = tmsg) {
- tmsg = msgs->next;
- free( msgs );
- }
- }
- #ifndef HAVE_VASPRINTF
- static int
- vasprintf( char **strp, const char *fmt, va_list ap )
- {
- int len;
- char tmp[1024];
- if ((len = vsnprintf( tmp, sizeof(tmp), fmt, ap )) < 0 || !(*strp = malloc( len + 1 )))
- return -1;
- if (len >= (int)sizeof(tmp))
- vsprintf( *strp, fmt, ap );
- else
- memcpy( *strp, tmp, len + 1 );
- return len;
- }
- #endif
- void
- oob( void )
- {
- fputs( "Fatal: buffer too small. Please report a bug.\n", stderr );
- abort();
- }
- int
- nfsnprintf( char *buf, int blen, const char *fmt, ... )
- {
- int ret;
- va_list va;
- va_start( va, fmt );
- if (blen <= 0 || (unsigned)(ret = vsnprintf( buf, blen, fmt, va )) >= (unsigned)blen)
- oob();
- va_end( va );
- return ret;
- }
- static void ATTR_NORETURN
- oom( void )
- {
- fputs( "Fatal: Out of memory\n", stderr );
- abort();
- }
- void *
- nfmalloc( size_t sz )
- {
- void *ret;
- if (!(ret = malloc( sz )))
- oom();
- return ret;
- }
- void *
- nfcalloc( size_t sz )
- {
- void *ret;
- if (!(ret = calloc( sz, 1 )))
- oom();
- return ret;
- }
- void *
- nfrealloc( void *mem, size_t sz )
- {
- char *ret;
- if (!(ret = realloc( mem, sz )) && sz)
- oom();
- return ret;
- }
- char *
- nfstrdup( const char *str )
- {
- char *ret;
- if (!(ret = strdup( str )))
- oom();
- return ret;
- }
- int
- nfvasprintf( char **str, const char *fmt, va_list va )
- {
- int ret = vasprintf( str, fmt, va );
- if (ret < 0)
- oom();
- return ret;
- }
- int
- nfasprintf( char **str, const char *fmt, ... )
- {
- int ret;
- va_list va;
- va_start( va, fmt );
- ret = nfvasprintf( str, fmt, va );
- va_end( va );
- return ret;
- }
- /*
- static struct passwd *
- cur_user( void )
- {
- char *p;
- struct passwd *pw;
- uid_t uid;
- uid = getuid();
- if ((!(p = getenv("LOGNAME")) || !(pw = getpwnam( p )) || pw->pw_uid != uid) &&
- (!(p = getenv("USER")) || !(pw = getpwnam( p )) || pw->pw_uid != uid) &&
- !(pw = getpwuid( uid )))
- {
- fputs ("Cannot determinate current user\n", stderr);
- return 0;
- }
- return pw;
- }
- */
- static char *
- my_strndup( const char *s, size_t nchars )
- {
- char *r = nfmalloc( nchars + 1 );
- memcpy( r, s, nchars );
- r[nchars] = 0;
- return r;
- }
- char *
- expand_strdup( const char *s )
- {
- struct passwd *pw;
- const char *p, *q;
- char *r;
- if (*s == '~') {
- s++;
- if (!*s) {
- p = 0;
- q = Home;
- } else if (*s == '/') {
- p = s;
- q = Home;
- } else {
- if ((p = strchr( s, '/' ))) {
- r = my_strndup( s, (int)(p - s) );
- pw = getpwnam( r );
- free( r );
- } else
- pw = getpwnam( s );
- if (!pw)
- return 0;
- q = pw->pw_dir;
- }
- nfasprintf( &r, "%s%s", q, p ? p : "" );
- return r;
- } else
- return nfstrdup( s );
- }
- static int
- compare_ints( const void *l, const void *r )
- {
- return *(int *)l - *(int *)r;
- }
- void
- sort_ints( int *arr, int len )
- {
- qsort( arr, len, sizeof(int), compare_ints );
- }
- static struct {
- unsigned char i, j, s[256];
- } rs;
- void
- arc4_init( void )
- {
- int i, fd;
- unsigned char j, si, dat[128];
- if ((fd = open( "/dev/urandom", O_RDONLY )) < 0 && (fd = open( "/dev/random", O_RDONLY )) < 0) {
- error( "Fatal: no random number source available.\n" );
- exit( 3 );
- }
- if (read( fd, dat, 128 ) != 128) {
- error( "Fatal: cannot read random number source.\n" );
- exit( 3 );
- }
- close( fd );
- for (i = 0; i < 256; i++)
- rs.s[i] = i;
- for (i = j = 0; i < 256; i++) {
- si = rs.s[i];
- j += si + dat[i & 127];
- rs.s[i] = rs.s[j];
- rs.s[j] = si;
- }
- rs.i = rs.j = 0;
- for (i = 0; i < 256; i++)
- arc4_getbyte();
- }
- unsigned char
- arc4_getbyte( void )
- {
- unsigned char si, sj;
- rs.i++;
- si = rs.s[rs.i];
- rs.j += si;
- sj = rs.s[rs.j];
- rs.s[rs.i] = sj;
- rs.s[rs.j] = si;
- return rs.s[(si + sj) & 0xff];
- }
|