util.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639
  1. /*
  2. * mbsync - mailbox synchronizer
  3. * Copyright (C) 2000-2002 Michael R. Elkins <me@mutt.org>
  4. * Copyright (C) 2002-2006,2011,2012 Oswald Buddenhagen <ossi@users.sf.net>
  5. *
  6. * This program is free software; you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License as published by
  8. * the Free Software Foundation; either version 2 of the License, or
  9. * (at your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. * GNU General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License
  17. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  18. *
  19. * As a special exception, mbsync may be linked with the OpenSSL library,
  20. * despite that library's more restrictive license.
  21. */
  22. #include "isync.h"
  23. #include <assert.h>
  24. #include <stdlib.h>
  25. #include <unistd.h>
  26. #include <fcntl.h>
  27. #include <string.h>
  28. #include <pwd.h>
  29. int DFlags;
  30. static int need_nl;
  31. void
  32. flushn( void )
  33. {
  34. if (need_nl) {
  35. putchar( '\n' );
  36. fflush( stdout );
  37. need_nl = 0;
  38. }
  39. }
  40. static void
  41. printn( const char *msg, va_list va )
  42. {
  43. if (*msg == '\v')
  44. msg++;
  45. else
  46. flushn();
  47. vprintf( msg, va );
  48. fflush( stdout );
  49. }
  50. void
  51. debug( const char *msg, ... )
  52. {
  53. va_list va;
  54. if (DFlags & DEBUG) {
  55. va_start( va, msg );
  56. vprintf( msg, va );
  57. va_end( va );
  58. fflush( stdout );
  59. need_nl = 0;
  60. }
  61. }
  62. void
  63. debugn( const char *msg, ... )
  64. {
  65. va_list va;
  66. if (DFlags & DEBUG) {
  67. va_start( va, msg );
  68. vprintf( msg, va );
  69. va_end( va );
  70. fflush( stdout );
  71. need_nl = 1;
  72. }
  73. }
  74. void
  75. info( const char *msg, ... )
  76. {
  77. va_list va;
  78. if (!(DFlags & QUIET)) {
  79. va_start( va, msg );
  80. printn( msg, va );
  81. va_end( va );
  82. need_nl = 0;
  83. }
  84. }
  85. void
  86. infon( const char *msg, ... )
  87. {
  88. va_list va;
  89. if (!(DFlags & QUIET)) {
  90. va_start( va, msg );
  91. printn( msg, va );
  92. va_end( va );
  93. need_nl = 1;
  94. }
  95. }
  96. void
  97. warn( const char *msg, ... )
  98. {
  99. va_list va;
  100. if (!(DFlags & VERYQUIET)) {
  101. flushn();
  102. va_start( va, msg );
  103. vfprintf( stderr, msg, va );
  104. va_end( va );
  105. }
  106. }
  107. void
  108. error( const char *msg, ... )
  109. {
  110. va_list va;
  111. flushn();
  112. va_start( va, msg );
  113. vfprintf( stderr, msg, va );
  114. va_end( va );
  115. }
  116. void
  117. sys_error( const char *msg, ... )
  118. {
  119. va_list va;
  120. char buf[1024];
  121. flushn();
  122. va_start( va, msg );
  123. if ((unsigned)vsnprintf( buf, sizeof(buf), msg, va ) >= sizeof(buf))
  124. oob();
  125. va_end( va );
  126. perror( buf );
  127. }
  128. void
  129. add_string_list( string_list_t **list, const char *str )
  130. {
  131. string_list_t *elem;
  132. int len;
  133. len = strlen( str );
  134. elem = nfmalloc( sizeof(*elem) + len );
  135. elem->next = *list;
  136. *list = elem;
  137. memcpy( elem->string, str, len + 1 );
  138. }
  139. void
  140. free_string_list( string_list_t *list )
  141. {
  142. string_list_t *tlist;
  143. for (; list; list = tlist) {
  144. tlist = list->next;
  145. free( list );
  146. }
  147. }
  148. void
  149. free_generic_messages( message_t *msgs )
  150. {
  151. message_t *tmsg;
  152. for (; msgs; msgs = tmsg) {
  153. tmsg = msgs->next;
  154. free( msgs );
  155. }
  156. }
  157. #ifndef HAVE_VASPRINTF
  158. static int
  159. vasprintf( char **strp, const char *fmt, va_list ap )
  160. {
  161. int len;
  162. char tmp[1024];
  163. if ((len = vsnprintf( tmp, sizeof(tmp), fmt, ap )) < 0 || !(*strp = malloc( len + 1 )))
  164. return -1;
  165. if (len >= (int)sizeof(tmp))
  166. vsprintf( *strp, fmt, ap );
  167. else
  168. memcpy( *strp, tmp, len + 1 );
  169. return len;
  170. }
  171. #endif
  172. #ifndef HAVE_MEMRCHR
  173. void *
  174. memrchr( const void *s, int c, size_t n )
  175. {
  176. u_char *b = (u_char *)s, *e = b + n;
  177. while (--e >= b)
  178. if (*e == c)
  179. return (void *)e;
  180. return 0;
  181. }
  182. #endif
  183. void
  184. oob( void )
  185. {
  186. fputs( "Fatal: buffer too small. Please report a bug.\n", stderr );
  187. abort();
  188. }
  189. int
  190. nfsnprintf( char *buf, int blen, const char *fmt, ... )
  191. {
  192. int ret;
  193. va_list va;
  194. va_start( va, fmt );
  195. if (blen <= 0 || (unsigned)(ret = vsnprintf( buf, blen, fmt, va )) >= (unsigned)blen)
  196. oob();
  197. va_end( va );
  198. return ret;
  199. }
  200. static void ATTR_NORETURN
  201. oom( void )
  202. {
  203. fputs( "Fatal: Out of memory\n", stderr );
  204. abort();
  205. }
  206. void *
  207. nfmalloc( size_t sz )
  208. {
  209. void *ret;
  210. if (!(ret = malloc( sz )))
  211. oom();
  212. return ret;
  213. }
  214. void *
  215. nfcalloc( size_t sz )
  216. {
  217. void *ret;
  218. if (!(ret = calloc( sz, 1 )))
  219. oom();
  220. return ret;
  221. }
  222. void *
  223. nfrealloc( void *mem, size_t sz )
  224. {
  225. char *ret;
  226. if (!(ret = realloc( mem, sz )) && sz)
  227. oom();
  228. return ret;
  229. }
  230. char *
  231. nfstrdup( const char *str )
  232. {
  233. char *ret;
  234. if (!(ret = strdup( str )))
  235. oom();
  236. return ret;
  237. }
  238. int
  239. nfvasprintf( char **str, const char *fmt, va_list va )
  240. {
  241. int ret = vasprintf( str, fmt, va );
  242. if (ret < 0)
  243. oom();
  244. return ret;
  245. }
  246. int
  247. nfasprintf( char **str, const char *fmt, ... )
  248. {
  249. int ret;
  250. va_list va;
  251. va_start( va, fmt );
  252. ret = nfvasprintf( str, fmt, va );
  253. va_end( va );
  254. return ret;
  255. }
  256. /*
  257. static struct passwd *
  258. cur_user( void )
  259. {
  260. char *p;
  261. struct passwd *pw;
  262. uid_t uid;
  263. uid = getuid();
  264. if ((!(p = getenv("LOGNAME")) || !(pw = getpwnam( p )) || pw->pw_uid != uid) &&
  265. (!(p = getenv("USER")) || !(pw = getpwnam( p )) || pw->pw_uid != uid) &&
  266. !(pw = getpwuid( uid )))
  267. {
  268. fputs ("Cannot determinate current user\n", stderr);
  269. return 0;
  270. }
  271. return pw;
  272. }
  273. */
  274. static char *
  275. my_strndup( const char *s, size_t nchars )
  276. {
  277. char *r = nfmalloc( nchars + 1 );
  278. memcpy( r, s, nchars );
  279. r[nchars] = 0;
  280. return r;
  281. }
  282. char *
  283. expand_strdup( const char *s )
  284. {
  285. struct passwd *pw;
  286. const char *p, *q;
  287. char *r;
  288. if (*s == '~') {
  289. s++;
  290. if (!*s) {
  291. rethome:
  292. p = 0;
  293. q = Home;
  294. } else if (*s == '/') {
  295. p = s + 1;
  296. q = Home;
  297. } else {
  298. if ((p = strchr( s, '/' ))) {
  299. r = my_strndup( s, (int)(p - s) );
  300. pw = getpwnam( r );
  301. free( r );
  302. p++;
  303. } else
  304. pw = getpwnam( s );
  305. if (!pw)
  306. return 0;
  307. q = pw->pw_dir;
  308. }
  309. if (!p)
  310. return nfstrdup( q );
  311. retjoin:
  312. nfasprintf( &r, "%s/%s", q, p );
  313. return r;
  314. } else if (*s != '/') {
  315. if (*s == '.' && !s[1])
  316. goto rethome;
  317. p = s;
  318. q = Home;
  319. goto retjoin;
  320. } else
  321. return nfstrdup( s );
  322. }
  323. /* Return value: 0 = ok, -1 = out found in arg, -2 = in found in arg but no out specified */
  324. int
  325. map_name( char *arg, char in, char out )
  326. {
  327. int l, k;
  328. if (!in || in == out)
  329. return 0;
  330. for (l = 0; arg[l]; l++)
  331. if (arg[l] == in) {
  332. if (!out)
  333. return -2;
  334. arg[l] = out;
  335. } else if (arg[l] == out) {
  336. /* restore original name for printing error message */
  337. for (k = 0; k < l; k++)
  338. if (arg[k] == out)
  339. arg[k] = in;
  340. return -1;
  341. }
  342. return 0;
  343. }
  344. static int
  345. compare_ints( const void *l, const void *r )
  346. {
  347. return *(int *)l - *(int *)r;
  348. }
  349. void
  350. sort_ints( int *arr, int len )
  351. {
  352. qsort( arr, len, sizeof(int), compare_ints );
  353. }
  354. static struct {
  355. unsigned char i, j, s[256];
  356. } rs;
  357. void
  358. arc4_init( void )
  359. {
  360. int i, fd;
  361. unsigned char j, si, dat[128];
  362. if ((fd = open( "/dev/urandom", O_RDONLY )) < 0 && (fd = open( "/dev/random", O_RDONLY )) < 0) {
  363. error( "Fatal: no random number source available.\n" );
  364. exit( 3 );
  365. }
  366. if (read( fd, dat, 128 ) != 128) {
  367. error( "Fatal: cannot read random number source.\n" );
  368. exit( 3 );
  369. }
  370. close( fd );
  371. for (i = 0; i < 256; i++)
  372. rs.s[i] = i;
  373. for (i = j = 0; i < 256; i++) {
  374. si = rs.s[i];
  375. j += si + dat[i & 127];
  376. rs.s[i] = rs.s[j];
  377. rs.s[j] = si;
  378. }
  379. rs.i = rs.j = 0;
  380. for (i = 0; i < 256; i++)
  381. arc4_getbyte();
  382. }
  383. unsigned char
  384. arc4_getbyte( void )
  385. {
  386. unsigned char si, sj;
  387. rs.i++;
  388. si = rs.s[rs.i];
  389. rs.j += si;
  390. sj = rs.s[rs.j];
  391. rs.s[rs.i] = sj;
  392. rs.s[rs.j] = si;
  393. return rs.s[(si + sj) & 0xff];
  394. }
  395. static const unsigned char prime_deltas[] = {
  396. 0, 0, 1, 3, 1, 5, 3, 3, 1, 9, 7, 5, 3, 9, 25, 3,
  397. 1, 21, 3, 21, 7, 15, 9, 5, 3, 29, 15, 0, 0, 0, 0, 0
  398. };
  399. int
  400. bucketsForSize( int size )
  401. {
  402. int base = 4, bits = 2;
  403. for (;;) {
  404. int prime = base + prime_deltas[bits];
  405. if (prime >= size)
  406. return prime;
  407. base <<= 1;
  408. bits++;
  409. }
  410. }
  411. #ifdef HAVE_SYS_POLL_H
  412. static struct pollfd *pollfds;
  413. #else
  414. # ifdef HAVE_SYS_SELECT_H
  415. # include <sys/select.h>
  416. # endif
  417. # define pollfds fdparms
  418. #endif
  419. static struct {
  420. void (*cb)( int what, void *aux );
  421. void *aux;
  422. #ifndef HAVE_SYS_POLL_H
  423. int fd, events;
  424. #endif
  425. int faked;
  426. } *fdparms;
  427. static int npolls, rpolls, changed;
  428. static int
  429. find_fd( int fd )
  430. {
  431. int n;
  432. for (n = 0; n < npolls; n++)
  433. if (pollfds[n].fd == fd)
  434. return n;
  435. return -1;
  436. }
  437. void
  438. add_fd( int fd, void (*cb)( int events, void *aux ), void *aux )
  439. {
  440. int n;
  441. assert( find_fd( fd ) < 0 );
  442. n = npolls++;
  443. if (rpolls < npolls) {
  444. rpolls = npolls;
  445. #ifdef HAVE_SYS_POLL_H
  446. pollfds = nfrealloc(pollfds, npolls * sizeof(*pollfds));
  447. #endif
  448. fdparms = nfrealloc(fdparms, npolls * sizeof(*fdparms));
  449. }
  450. pollfds[n].fd = fd;
  451. pollfds[n].events = 0; /* POLLERR & POLLHUP implicit */
  452. fdparms[n].faked = 0;
  453. fdparms[n].cb = cb;
  454. fdparms[n].aux = aux;
  455. changed = 1;
  456. }
  457. void
  458. conf_fd( int fd, int and_events, int or_events )
  459. {
  460. int n = find_fd( fd );
  461. assert( n >= 0 );
  462. pollfds[n].events = (pollfds[n].events & and_events) | or_events;
  463. }
  464. void
  465. fake_fd( int fd, int events )
  466. {
  467. int n = find_fd( fd );
  468. assert( n >= 0 );
  469. fdparms[n].faked |= events;
  470. }
  471. void
  472. del_fd( int fd )
  473. {
  474. int n = find_fd( fd );
  475. assert( n >= 0 );
  476. npolls--;
  477. #ifdef HAVE_SYS_POLL_H
  478. memmove(pollfds + n, pollfds + n + 1, (npolls - n) * sizeof(*pollfds));
  479. #endif
  480. memmove(fdparms + n, fdparms + n + 1, (npolls - n) * sizeof(*fdparms));
  481. changed = 1;
  482. }
  483. #define shifted_bit(in, from, to) \
  484. (((unsigned)(in) & from) \
  485. / (from > to ? from / to : 1) \
  486. * (to > from ? to / from : 1))
  487. static void
  488. event_wait( void )
  489. {
  490. int m, n;
  491. #ifdef HAVE_SYS_POLL_H
  492. int timeout = -1;
  493. for (n = 0; n < npolls; n++)
  494. if (fdparms[n].faked) {
  495. timeout = 0;
  496. break;
  497. }
  498. if (poll( pollfds, npolls, timeout ) < 0) {
  499. perror( "poll() failed in event loop" );
  500. abort();
  501. }
  502. for (n = 0; n < npolls; n++)
  503. if ((m = pollfds[n].revents | fdparms[n].faked)) {
  504. assert( !(m & POLLNVAL) );
  505. fdparms[n].faked = 0;
  506. fdparms[n].cb( m | shifted_bit( m, POLLHUP, POLLIN ), fdparms[n].aux );
  507. if (changed) {
  508. changed = 0;
  509. break;
  510. }
  511. }
  512. #else
  513. struct timeval *timeout = 0;
  514. static struct timeval null_tv;
  515. fd_set rfds, wfds, efds;
  516. int fd;
  517. FD_ZERO( &rfds );
  518. FD_ZERO( &wfds );
  519. FD_ZERO( &efds );
  520. m = -1;
  521. for (n = 0; n < npolls; n++) {
  522. if (fdparms[n].faked)
  523. timeout = &null_tv;
  524. fd = fdparms[n].fd;
  525. if (fdparms[n].events & POLLIN)
  526. FD_SET( fd, &rfds );
  527. if (fdparms[n].events & POLLOUT)
  528. FD_SET( fd, &wfds );
  529. FD_SET( fd, &efds );
  530. if (fd > m)
  531. m = fd;
  532. }
  533. if (select( m + 1, &rfds, &wfds, &efds, timeout ) < 0) {
  534. perror( "select() failed in event loop" );
  535. abort();
  536. }
  537. for (n = 0; n < npolls; n++) {
  538. fd = fdparms[n].fd;
  539. m = fdparms[n].faked;
  540. if (FD_ISSET( fd, &rfds ))
  541. m |= POLLIN;
  542. if (FD_ISSET( fd, &wfds ))
  543. m |= POLLOUT;
  544. if (FD_ISSET( fd, &efds ))
  545. m |= POLLERR;
  546. if (m) {
  547. fdparms[n].faked = 0;
  548. fdparms[n].cb( m, fdparms[n].aux );
  549. if (changed) {
  550. changed = 0;
  551. break;
  552. }
  553. }
  554. }
  555. #endif
  556. }
  557. void
  558. main_loop( void )
  559. {
  560. while (npolls)
  561. event_wait();
  562. }