18e25f19bSMatthew Dillon /* 28e25f19bSMatthew Dillon * Copyright (c) 2019 The DragonFly Project. All rights reserved. 38e25f19bSMatthew Dillon * 48e25f19bSMatthew Dillon * This code is derived from software contributed to The DragonFly Project 58e25f19bSMatthew Dillon * by Matthew Dillon <dillon@backplane.com> 68e25f19bSMatthew Dillon * 78e25f19bSMatthew Dillon * This code uses concepts and configuration based on 'synth', by 88e25f19bSMatthew Dillon * John R. Marino <draco@marino.st>, which was written in ada. 98e25f19bSMatthew Dillon * 108e25f19bSMatthew Dillon * Redistribution and use in source and binary forms, with or without 118e25f19bSMatthew Dillon * modification, are permitted provided that the following conditions 128e25f19bSMatthew Dillon * are met: 138e25f19bSMatthew Dillon * 148e25f19bSMatthew Dillon * 1. Redistributions of source code must retain the above copyright 158e25f19bSMatthew Dillon * notice, this list of conditions and the following disclaimer. 168e25f19bSMatthew Dillon * 2. Redistributions in binary form must reproduce the above copyright 178e25f19bSMatthew Dillon * notice, this list of conditions and the following disclaimer in 188e25f19bSMatthew Dillon * the documentation and/or other materials provided with the 198e25f19bSMatthew Dillon * distribution. 208e25f19bSMatthew Dillon * 3. Neither the name of The DragonFly Project nor the names of its 218e25f19bSMatthew Dillon * contributors may be used to endorse or promote products derived 228e25f19bSMatthew Dillon * from this software without specific, prior written permission. 238e25f19bSMatthew Dillon * 248e25f19bSMatthew Dillon * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 258e25f19bSMatthew Dillon * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 268e25f19bSMatthew Dillon * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 278e25f19bSMatthew Dillon * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE 288e25f19bSMatthew Dillon * COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 298e25f19bSMatthew Dillon * INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING, 308e25f19bSMatthew Dillon * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 318e25f19bSMatthew Dillon * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 328e25f19bSMatthew Dillon * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 338e25f19bSMatthew Dillon * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT 348e25f19bSMatthew Dillon * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 358e25f19bSMatthew Dillon * SUCH DAMAGE. 368e25f19bSMatthew Dillon */ 378e25f19bSMatthew Dillon 388e25f19bSMatthew Dillon #include <sys/types.h> 398e25f19bSMatthew Dillon #include <sys/wait.h> 408e25f19bSMatthew Dillon #include <sys/stat.h> 418e25f19bSMatthew Dillon #include <sys/sysctl.h> 428e25f19bSMatthew Dillon #include <sys/socket.h> 438e25f19bSMatthew Dillon #include <sys/mount.h> 448e25f19bSMatthew Dillon #include <sys/procctl.h> 45*32fc5bbfSMatthew Dillon #if defined(__DragonFly__) 46*32fc5bbfSMatthew Dillon #include <sys/vmmeter.h> 47*32fc5bbfSMatthew Dillon #endif 488e25f19bSMatthew Dillon #include <stdio.h> 498e25f19bSMatthew Dillon #include <stdlib.h> 508e25f19bSMatthew Dillon #include <stddef.h> 518e25f19bSMatthew Dillon #include <stdarg.h> 528e25f19bSMatthew Dillon #include <unistd.h> 538e25f19bSMatthew Dillon #include <string.h> 548e25f19bSMatthew Dillon #include <fcntl.h> 558e25f19bSMatthew Dillon #include <signal.h> 568e25f19bSMatthew Dillon #include <poll.h> 578e25f19bSMatthew Dillon #include <assert.h> 588e25f19bSMatthew Dillon #include <errno.h> 598e25f19bSMatthew Dillon #include <pthread.h> 608e25f19bSMatthew Dillon #include <dirent.h> 618e25f19bSMatthew Dillon #include <termios.h> 628b485838SMatthew Dillon #include <time.h> 638e25f19bSMatthew Dillon #include <ctype.h> 648e25f19bSMatthew Dillon 6568dc2eeaSMatthew Dillon /* 6668dc2eeaSMatthew Dillon * More esoteric headers 6768dc2eeaSMatthew Dillon */ 6868dc2eeaSMatthew Dillon #include <libutil.h> /* forkpty() */ 6968dc2eeaSMatthew Dillon #include <arpa/inet.h> /* ntohl() */ 7068dc2eeaSMatthew Dillon #include <elf.h> /* try to get elf info */ 71a574cbf0SMatthew Dillon 728e25f19bSMatthew Dillon struct pkglink; 738e25f19bSMatthew Dillon 74b1224b54SMatthew Dillon #define DSYNTH_VERSION "1.02" 758e25f19bSMatthew Dillon #define MAXWORKERS 1024 76aac7a6d9SMatthew Dillon #define MAXLOGLINES 1024 778e25f19bSMatthew Dillon #define MAXJOBS 8192 /* just used for -j sanity */ 788e25f19bSMatthew Dillon #define MAXBULK MAXWORKERS 798e25f19bSMatthew Dillon 808e25f19bSMatthew Dillon #define MAKE_BINARY "/usr/bin/make" 818e25f19bSMatthew Dillon #define PKG_BINARY "/usr/local/sbin/pkg" 828e25f19bSMatthew Dillon #define MOUNT_BINARY "/sbin/mount" 838e25f19bSMatthew Dillon #define MOUNT_NULLFS_BINARY "/sbin/mount_null" 848e25f19bSMatthew Dillon #define MOUNT_TMPFS_BINARY "/sbin/mount_tmpfs" 858e25f19bSMatthew Dillon #define MOUNT_DEVFS_BINARY "/sbin/mount_devfs" 868ec23ca1SMatthew Dillon #define MOUNT_PROCFS_BINARY "/sbin/mount_procfs" 878e25f19bSMatthew Dillon #define UMOUNT_BINARY "/sbin/umount" 888e25f19bSMatthew Dillon 89aac7a6d9SMatthew Dillon #define STATS_FILE "monitor.dat" /* under LogsPath */ 90aac7a6d9SMatthew Dillon #define STATS_LOCKFILE "monitor.lk" /* under LogsPath */ 91aac7a6d9SMatthew Dillon 927f0eca56SMatthew Dillon #define ONEGB (1024L * 1024 * 1024) 93f4094b20SMatthew Dillon #define DISABLED_STR "disabled" 947f0eca56SMatthew Dillon 958e25f19bSMatthew Dillon /* 96667fb2cbSMatthew Dillon * This can be ".tar", ".tgz", ".txz", or ".tbz". 97667fb2cbSMatthew Dillon * 98667fb2cbSMatthew Dillon * .tar - very fast but you'll need 1TB+ of storage just for the package files. 99667fb2cbSMatthew Dillon * .txz - very compact but decompression speed is horrible. 100667fb2cbSMatthew Dillon * .tgz - reasonable compression, extremely fast decompression. Roughly 101667fb2cbSMatthew Dillon * 1.1x to 2.0x the size of a .txz, but decompresses 10x faster. 102667fb2cbSMatthew Dillon * .tbz - worse than .tgz generally 103f17a90cbSMatthew Dillon * 104f17a90cbSMatthew Dillon * NOTE: Decompression speed does effect bulk builds since each slot has 105f17a90cbSMatthew Dillon * to install pre-reqs before building any particular package. Set 106f17a90cbSMatthew Dillon * the default to .txz to remain close to synth's default. 107667fb2cbSMatthew Dillon */ 108f17a90cbSMatthew Dillon #define USE_PKG_SUFX ".txz" 109667fb2cbSMatthew Dillon 110667fb2cbSMatthew Dillon /* 1118e25f19bSMatthew Dillon * Topology linkages 1128e25f19bSMatthew Dillon */ 1138e25f19bSMatthew Dillon typedef struct pkglink { 1148e25f19bSMatthew Dillon struct pkglink *next; 1158e25f19bSMatthew Dillon struct pkglink *prev; 1168e25f19bSMatthew Dillon struct pkg *pkg; 11763fcce5bSMatthew Dillon int dep_type; 1188e25f19bSMatthew Dillon } pkglink_t; 1198e25f19bSMatthew Dillon 12063fcce5bSMatthew Dillon #define DEP_TYPE_FETCH 1 12163fcce5bSMatthew Dillon #define DEP_TYPE_EXT 2 12263fcce5bSMatthew Dillon #define DEP_TYPE_PATCH 3 12363fcce5bSMatthew Dillon #define DEP_TYPE_BUILD 4 12463fcce5bSMatthew Dillon #define DEP_TYPE_LIB 5 12563fcce5bSMatthew Dillon #define DEP_TYPE_RUN 6 12663fcce5bSMatthew Dillon 1278e25f19bSMatthew Dillon /* 1288e25f19bSMatthew Dillon * Describes a [flavored] package 1298e25f19bSMatthew Dillon */ 1308e25f19bSMatthew Dillon typedef struct pkg { 1318e25f19bSMatthew Dillon struct pkg *build_next; /* topology inversion build list */ 1328e25f19bSMatthew Dillon struct pkg *bnext; /* linked list from bulk return */ 1338e25f19bSMatthew Dillon struct pkg *hnext1; /* hash based on portdir */ 1348e25f19bSMatthew Dillon struct pkg *hnext2; /* hash based on pkgfile */ 1358e25f19bSMatthew Dillon pkglink_t idepon_list; /* I need these pkgs */ 1368e25f19bSMatthew Dillon pkglink_t deponi_list; /* pkgs which depend on me */ 1378e25f19bSMatthew Dillon char *portdir; /* origin name e.g. www/chromium[@flavor] */ 1388e25f19bSMatthew Dillon char *logfile; /* relative logfile path */ 1398e25f19bSMatthew Dillon char *version; /* PKGVERSION - e.g. 3.5.0_1 */ 1408e25f19bSMatthew Dillon char *pkgfile; /* PKGFILE - e.g. flav-blah-3.5.0_1.txz */ 1411645cafeSMatthew Dillon char *distfiles; /* DISTFILES - e.g. blah-68.0.source.tar.xz */ 1421645cafeSMatthew Dillon char *distsubdir; /* DIST_SUBDIR- e.g. cabal */ 14387017ac4SMatthew Dillon char *ignore; /* IGNORE (also covers BROKEN) */ 1448e25f19bSMatthew Dillon char *fetch_deps; /* FETCH_DEPENDS */ 1458e25f19bSMatthew Dillon char *ext_deps; /* EXTRACT_DEPENDS */ 1468e25f19bSMatthew Dillon char *patch_deps; /* PATCH_DEPENDS */ 1478e25f19bSMatthew Dillon char *build_deps; /* BUILD_DEPENDS */ 1488e25f19bSMatthew Dillon char *lib_deps; /* LIB_DEPENDS */ 1498e25f19bSMatthew Dillon char *run_deps; /* RUN_DEPENDS */ 1508e25f19bSMatthew Dillon char *pos_options; /* SELECTED_OPTIONS */ 1518e25f19bSMatthew Dillon char *neg_options; /* DESELECTED_OPTIONS */ 1528e25f19bSMatthew Dillon char *flavors; /* FLAVORS - e.g. py36 py27 */ 153f4094b20SMatthew Dillon char *uses; /* USES (metaport test) */ 1548e25f19bSMatthew Dillon int make_jobs_number; /* MAKE_JOBS_NUMBER */ 1558e25f19bSMatthew Dillon int use_linux; /* USE_LINUX */ 1568e25f19bSMatthew Dillon int idep_count; /* count recursive idepon build deps */ 1578e25f19bSMatthew Dillon int depi_count; /* count recursive deponi build deps */ 15888c24d72SMatthew Dillon int depi_depth; /* tree depth who depends on me */ 1598e25f19bSMatthew Dillon int dsynth_install_flg; /* locked with WorkerMutex */ 1608e25f19bSMatthew Dillon int flags; 1613dd48cfaSMatthew Dillon int rscan; /* recursive scan flag (serialized use) */ 162fef2fc63SMatthew Dillon size_t pkgfile_size; /* size of pkgfile */ 1638e25f19bSMatthew Dillon } pkg_t; 1648e25f19bSMatthew Dillon 1658e25f19bSMatthew Dillon #define PKGF_PACKAGED 0x00000001 /* has a repo package */ 1668e25f19bSMatthew Dillon #define PKGF_DUMMY 0x00000002 /* generic root for flavors */ 1678e25f19bSMatthew Dillon #define PKGF_NOTFOUND 0x00000004 /* dport not found */ 1688e25f19bSMatthew Dillon #define PKGF_CORRUPT 0x00000008 /* dport corrupt */ 1698e25f19bSMatthew Dillon #define PKGF_PLACEHOLD 0x00000010 /* pre-entered */ 1708e25f19bSMatthew Dillon #define PKGF_BUILDLIST 0x00000020 /* on build_list */ 1718e25f19bSMatthew Dillon #define PKGF_BUILDLOOP 0x00000040 /* traversal loop test */ 1728e25f19bSMatthew Dillon #define PKGF_BUILDTRAV 0x00000080 /* traversal optimization */ 1738e25f19bSMatthew Dillon #define PKGF_NOBUILD_D 0x00000100 /* can't build - dependency problem */ 1748e25f19bSMatthew Dillon #define PKGF_NOBUILD_S 0x00000200 /* can't build - skipped */ 1758e25f19bSMatthew Dillon #define PKGF_NOBUILD_F 0x00000400 /* can't build - failed */ 17687017ac4SMatthew Dillon #define PKGF_NOBUILD_I 0x00000800 /* can't build - ignored or broken */ 17787017ac4SMatthew Dillon #define PKGF_SUCCESS 0x00001000 /* build complete */ 17887017ac4SMatthew Dillon #define PKGF_FAILURE 0x00002000 /* build complete */ 17987017ac4SMatthew Dillon #define PKGF_RUNNING 0x00004000 /* build complete */ 18087017ac4SMatthew Dillon #define PKGF_PKGPKG 0x00008000 /* pkg/pkg-static special */ 18187017ac4SMatthew Dillon #define PKGF_NOTREADY 0x00010000 /* build_find_leaves() only */ 182f4094b20SMatthew Dillon #define PKGF_MANUALSEL 0x00020000 /* manually specified */ 183f4094b20SMatthew Dillon #define PKGF_META 0x00040000 /* USES contains 'metaport' */ 184710838f7SMatthew Dillon #define PKGF_DEBUGSTOP 0x00080000 /* freeze slot on completion */ 185710838f7SMatthew Dillon 1868e25f19bSMatthew Dillon #define PKGF_ERROR (PKGF_PLACEHOLD | PKGF_CORRUPT | PKGF_NOTFOUND | \ 1878e25f19bSMatthew Dillon PKGF_FAILURE) 18887017ac4SMatthew Dillon #define PKGF_NOBUILD (PKGF_NOBUILD_D | PKGF_NOBUILD_S | PKGF_NOBUILD_F | \ 18987017ac4SMatthew Dillon PKGF_NOBUILD_I) 1908e25f19bSMatthew Dillon 1918e25f19bSMatthew Dillon #define PKGLIST_EMPTY(pkglink) ((pkglink)->next == (pkglink)) 1928e25f19bSMatthew Dillon #define PKGLIST_FOREACH(var, head) \ 1938e25f19bSMatthew Dillon for (var = (head)->next; var != (head); var = (var)->next) 1948e25f19bSMatthew Dillon 1958e25f19bSMatthew Dillon typedef struct bulk { 1968e25f19bSMatthew Dillon struct bulk *next; 1978e25f19bSMatthew Dillon pthread_t td; 1988e25f19bSMatthew Dillon int debug; 1998e25f19bSMatthew Dillon int flags; 2008e25f19bSMatthew Dillon enum { UNLISTED, ONSUBMIT, ONRUN, ISRUNNING, ONRESPONSE } state; 2018e25f19bSMatthew Dillon char *s1; 2028e25f19bSMatthew Dillon char *s2; 2038e25f19bSMatthew Dillon char *s3; 2048e25f19bSMatthew Dillon char *s4; 2058e25f19bSMatthew Dillon char *r1; 2068e25f19bSMatthew Dillon char *r2; 2078e25f19bSMatthew Dillon char *r3; 2088e25f19bSMatthew Dillon char *r4; 2098e25f19bSMatthew Dillon pkg_t *list; /* pkgs linked by bnext */ 2108e25f19bSMatthew Dillon } bulk_t; 2118e25f19bSMatthew Dillon 2128e25f19bSMatthew Dillon /* 2138e25f19bSMatthew Dillon * Worker state (up to MAXWORKERS). Each worker operates within a 2148e25f19bSMatthew Dillon * chroot or jail. A system mirror is setup and the template 2158e25f19bSMatthew Dillon * is copied in. 2168e25f19bSMatthew Dillon * 2178e25f19bSMatthew Dillon * basedir - tmpfs 2188e25f19bSMatthew Dillon * /bin - nullfs (ro) 2198e25f19bSMatthew Dillon * /sbin - nullfs (ro) 2208e25f19bSMatthew Dillon * /lib - nullfs (ro) 2218e25f19bSMatthew Dillon * /libexec - nullfs (ro) 2228e25f19bSMatthew Dillon * /usr/bin - nullfs (ro) 2238e25f19bSMatthew Dillon * /usr/include - nullfs (ro) 2248e25f19bSMatthew Dillon * /usr/lib - nullfs (ro) 2258e25f19bSMatthew Dillon * /usr/libdata - nullfs (ro) 2268e25f19bSMatthew Dillon * /usr/libexec - nullfs (ro) 2278e25f19bSMatthew Dillon * /usr/sbin - nullfs (ro) 2288e25f19bSMatthew Dillon * /usr/share - nullfs (ro) 2298e25f19bSMatthew Dillon * /xports - nullfs (ro) 2308e25f19bSMatthew Dillon * /options - nullfs (ro) 2318e25f19bSMatthew Dillon * /packages - nullfs (ro) 2328e25f19bSMatthew Dillon * /distfiles - nullfs (ro) 2338e25f19bSMatthew Dillon * construction - tmpfs 2348e25f19bSMatthew Dillon * /usr/local - tmpfs 2358e25f19bSMatthew Dillon * /boot - nullfs (ro) 2368e25f19bSMatthew Dillon * /boot/modules.local - tmpfs 2378e25f19bSMatthew Dillon * /usr/games - nullfs (ro) 2388e25f19bSMatthew Dillon * /usr/src - nullfs (ro) 2398e25f19bSMatthew Dillon * /dev - devfs 2408e25f19bSMatthew Dillon */ 2418e25f19bSMatthew Dillon enum worker_state { WORKER_NONE, WORKER_IDLE, WORKER_PENDING, 2428e25f19bSMatthew Dillon WORKER_RUNNING, WORKER_DONE, WORKER_FAILED, 2438ec23ca1SMatthew Dillon WORKER_FROZEN, WORKER_EXITING }; 2448e25f19bSMatthew Dillon typedef enum worker_state worker_state_t; 2458e25f19bSMatthew Dillon 2468e25f19bSMatthew Dillon enum worker_phase { PHASE_PENDING, 2478e25f19bSMatthew Dillon PHASE_INSTALL_PKGS, 2488e25f19bSMatthew Dillon PHASE_CHECK_SANITY, 2498e25f19bSMatthew Dillon PHASE_PKG_DEPENDS, 2508e25f19bSMatthew Dillon PHASE_FETCH_DEPENDS, 2518e25f19bSMatthew Dillon PHASE_FETCH, 2528e25f19bSMatthew Dillon PHASE_CHECKSUM, 2538e25f19bSMatthew Dillon PHASE_EXTRACT_DEPENDS, 2548e25f19bSMatthew Dillon PHASE_EXTRACT, 2558e25f19bSMatthew Dillon PHASE_PATCH_DEPENDS, 2568e25f19bSMatthew Dillon PHASE_PATCH, 2578e25f19bSMatthew Dillon PHASE_BUILD_DEPENDS, 2588e25f19bSMatthew Dillon PHASE_LIB_DEPENDS, 2598e25f19bSMatthew Dillon PHASE_CONFIGURE, 2608e25f19bSMatthew Dillon PHASE_BUILD, 2618e25f19bSMatthew Dillon PHASE_RUN_DEPENDS, 2628e25f19bSMatthew Dillon PHASE_STAGE, 2638e25f19bSMatthew Dillon PHASE_TEST, 2648e25f19bSMatthew Dillon PHASE_CHECK_PLIST, 2658e25f19bSMatthew Dillon PHASE_PACKAGE, 2668e25f19bSMatthew Dillon PHASE_INSTALL_MTREE, 2678e25f19bSMatthew Dillon PHASE_INSTALL, 2688e25f19bSMatthew Dillon PHASE_DEINSTALL 2698e25f19bSMatthew Dillon }; 2708e25f19bSMatthew Dillon 2718e25f19bSMatthew Dillon typedef enum worker_phase worker_phase_t; 2728e25f19bSMatthew Dillon 2738e25f19bSMatthew Dillon /* 2748e25f19bSMatthew Dillon * Watchdog timeouts, in minutes, baseline, scales up with load/ncpus but 2758e25f19bSMatthew Dillon * does not scale down. 2768e25f19bSMatthew Dillon */ 2778e25f19bSMatthew Dillon #define WDOG1 (5) 2788e25f19bSMatthew Dillon #define WDOG2 (10) 2798e25f19bSMatthew Dillon #define WDOG3 (15) 2808e25f19bSMatthew Dillon #define WDOG4 (30) 2818e25f19bSMatthew Dillon #define WDOG5 (60) 2828e25f19bSMatthew Dillon #define WDOG6 (60 + 30) 2838e25f19bSMatthew Dillon #define WDOG7 (60 * 2) 2848e25f19bSMatthew Dillon #define WDOG8 (60 * 2 + 30) 2858e25f19bSMatthew Dillon #define WDOG9 (60 * 3) 2868e25f19bSMatthew Dillon 2878e25f19bSMatthew Dillon typedef struct worker { 2888e25f19bSMatthew Dillon int index; /* worker number 0..N-1 */ 2898e25f19bSMatthew Dillon int flags; 2908e25f19bSMatthew Dillon int accum_error; /* cumulative error */ 2918e25f19bSMatthew Dillon int mount_error; /* mount and unmount error */ 2928e25f19bSMatthew Dillon int terminate : 1; /* request sub-thread to terminate */ 2938e25f19bSMatthew Dillon char *basedir; /* base directory including id */ 2948e25f19bSMatthew Dillon char *flavor; 2958e25f19bSMatthew Dillon pthread_t td; /* pthread */ 2968e25f19bSMatthew Dillon pthread_cond_t cond; /* interlock cond (w/ WorkerMutex) */ 2978e25f19bSMatthew Dillon pkg_t *pkg; 2988e25f19bSMatthew Dillon worker_state_t state; /* general worker state */ 2998e25f19bSMatthew Dillon worker_phase_t phase; /* phase control in childBuilderThread */ 3008e25f19bSMatthew Dillon time_t start_time; 3018e25f19bSMatthew Dillon long lines; 302516819d9SMatthew Dillon long memuse; 3038e25f19bSMatthew Dillon pid_t pid; 3048e25f19bSMatthew Dillon int fds[2]; /* forked environment process */ 3058e25f19bSMatthew Dillon char status[64]; 306fef2fc63SMatthew Dillon size_t pkg_dep_size; /* pkg dependency size(s) */ 3078e25f19bSMatthew Dillon } worker_t; 3088e25f19bSMatthew Dillon 3098e25f19bSMatthew Dillon #define WORKERF_STATUS_UPDATE 0x0001 /* display update */ 3108e25f19bSMatthew Dillon #define WORKERF_SUCCESS 0x0002 /* completion flag */ 3118e25f19bSMatthew Dillon #define WORKERF_FAILURE 0x0004 /* completion flag */ 3128ec23ca1SMatthew Dillon #define WORKERF_FREEZE 0x0008 /* freeze the worker */ 3138e25f19bSMatthew Dillon 3148e25f19bSMatthew Dillon #define MOUNT_TYPE_MASK 0x000F 3158e25f19bSMatthew Dillon #define MOUNT_TYPE_TMPFS 0x0001 3168e25f19bSMatthew Dillon #define MOUNT_TYPE_NULLFS 0x0002 3178e25f19bSMatthew Dillon #define MOUNT_TYPE_DEVFS 0x0003 3188ec23ca1SMatthew Dillon #define MOUNT_TYPE_PROCFS 0x0004 3198e25f19bSMatthew Dillon #define MOUNT_TYPE_RW 0x0010 3208e25f19bSMatthew Dillon #define MOUNT_TYPE_BIG 0x0020 3218e25f19bSMatthew Dillon #define MOUNT_TYPE_TMP 0x0040 3228e25f19bSMatthew Dillon 3238e25f19bSMatthew Dillon #define NULLFS_RO (MOUNT_TYPE_NULLFS) 3248e25f19bSMatthew Dillon #define NULLFS_RW (MOUNT_TYPE_NULLFS | MOUNT_TYPE_RW) 3258ec23ca1SMatthew Dillon #define PROCFS_RO (MOUNT_TYPE_PROCFS) 3268e25f19bSMatthew Dillon #define TMPFS_RW (MOUNT_TYPE_TMPFS | MOUNT_TYPE_RW) 3278e25f19bSMatthew Dillon #define TMPFS_RW_BIG (MOUNT_TYPE_TMPFS | MOUNT_TYPE_RW | \ 3288e25f19bSMatthew Dillon MOUNT_TYPE_BIG) 3298e25f19bSMatthew Dillon #define DEVFS_RW (MOUNT_TYPE_DEVFS | MOUNT_TYPE_RW) 3308e25f19bSMatthew Dillon 3318e25f19bSMatthew Dillon /* 3328e25f19bSMatthew Dillon * IPC messages between the worker support thread and the worker process. 3338e25f19bSMatthew Dillon */ 3348e25f19bSMatthew Dillon typedef struct wmsg { 3358e25f19bSMatthew Dillon int cmd; 3368e25f19bSMatthew Dillon int status; 3378e25f19bSMatthew Dillon long lines; 338516819d9SMatthew Dillon long memuse; 3398e25f19bSMatthew Dillon worker_phase_t phase; 3408e25f19bSMatthew Dillon } wmsg_t; 3418e25f19bSMatthew Dillon 3428e25f19bSMatthew Dillon #define WMSG_CMD_STATUS_UPDATE 0x0001 3438e25f19bSMatthew Dillon #define WMSG_CMD_SUCCESS 0x0002 3448e25f19bSMatthew Dillon #define WMSG_CMD_FAILURE 0x0003 3458e25f19bSMatthew Dillon #define WMSG_CMD_INSTALL_PKGS 0x0004 3468e25f19bSMatthew Dillon #define WMSG_RES_INSTALL_PKGS 0x0005 3478ec23ca1SMatthew Dillon #define WMSG_CMD_FREEZEWORKER 0x0006 3488e25f19bSMatthew Dillon 3498e25f19bSMatthew Dillon /* 3508e25f19bSMatthew Dillon * Make variables and build environment 3518e25f19bSMatthew Dillon */ 3528e25f19bSMatthew Dillon typedef struct buildenv { 3538e25f19bSMatthew Dillon struct buildenv *next; 35468dc2eeaSMatthew Dillon const char *label; 35568dc2eeaSMatthew Dillon const char *data; 35668dc2eeaSMatthew Dillon char *a1; /* allocations */ 35768dc2eeaSMatthew Dillon char *a2; /* allocations */ 3586fd67931SMatthew Dillon int type; 3598e25f19bSMatthew Dillon } buildenv_t; 3608e25f19bSMatthew Dillon 3618e25f19bSMatthew Dillon /* 3628e25f19bSMatthew Dillon * Operating systems recognized by dsynth 3638e25f19bSMatthew Dillon */ 3648e25f19bSMatthew Dillon enum os_id { 3658e25f19bSMatthew Dillon OS_UNKNOWN, OS_DRAGONFLY, OS_FREEBSD, OS_NETBSD, OS_LINUX 3668e25f19bSMatthew Dillon }; 3678e25f19bSMatthew Dillon 3688e25f19bSMatthew Dillon typedef enum os_id os_id_t; 3698e25f19bSMatthew Dillon 3708e25f19bSMatthew Dillon /* 3718e25f19bSMatthew Dillon * DLOG 3728e25f19bSMatthew Dillon */ 3738e25f19bSMatthew Dillon #define DLOG_ALL 0 /* Usually stdout when curses disabled */ 3748e25f19bSMatthew Dillon #define DLOG_SUCC 1 /* success_list.log */ 3758e25f19bSMatthew Dillon #define DLOG_FAIL 2 /* failure_list.log */ 3768e25f19bSMatthew Dillon #define DLOG_IGN 3 /* ignored_list.log */ 3778e25f19bSMatthew Dillon #define DLOG_SKIP 4 /* skipped_list.log */ 3788e25f19bSMatthew Dillon #define DLOG_ABN 5 /* abnormal_command_output */ 3798e25f19bSMatthew Dillon #define DLOG_OBS 6 /* obsolete_packages.log */ 3808e25f19bSMatthew Dillon #define DLOG_COUNT 7 /* total number of DLOGs */ 381a574cbf0SMatthew Dillon #define DLOG_MASK 0x0FF 382a574cbf0SMatthew Dillon 383a574cbf0SMatthew Dillon #define DLOG_FILTER 0x100 /* Filter out of stdout in non-curses mode */ 384a574cbf0SMatthew Dillon #define DLOG_RED 0x200 /* Print in color */ 385a574cbf0SMatthew Dillon #define DLOG_GRN 0x400 /* Print in color */ 38633170e2dSMatthew Dillon #define DLOG_STDOUT 0x800 /* And stdout */ 3878e25f19bSMatthew Dillon 3888e25f19bSMatthew Dillon #define dassert(exp, fmt, ...) \ 3898e25f19bSMatthew Dillon if (!(exp)) dpanic(fmt, ## __VA_ARGS__) 3908e25f19bSMatthew Dillon 3918e25f19bSMatthew Dillon #define ddassert(exp) \ 3928e25f19bSMatthew Dillon dassert((exp), "\"%s\" line %d", __FILE__, __LINE__) 3938e25f19bSMatthew Dillon 3948e25f19bSMatthew Dillon #define dassert_errno(exp, fmt, ...) \ 3959c4c701fSMatthew Dillon if (!(exp)) dpanic_errno(fmt, ## __VA_ARGS__) 3968e25f19bSMatthew Dillon 3978e25f19bSMatthew Dillon #define dlog(which, fmt, ...) \ 3988e25f19bSMatthew Dillon _dlog(which, fmt, ## __VA_ARGS__) 3998e25f19bSMatthew Dillon 400a574cbf0SMatthew Dillon #define dlog_tsnl(which, fmt, ...) \ 401a574cbf0SMatthew Dillon _dlog(which, fmt, ## __VA_ARGS__) 402a574cbf0SMatthew Dillon 4038e25f19bSMatthew Dillon #define dfatal(fmt, ...) \ 4048e25f19bSMatthew Dillon _dfatal(__FILE__, __LINE__, __func__, 0, fmt, ## __VA_ARGS__) 4058e25f19bSMatthew Dillon 4068e25f19bSMatthew Dillon #define dpanic(fmt, ...) \ 4078e25f19bSMatthew Dillon _dfatal(__FILE__, __LINE__, __func__, 2, fmt, ## __VA_ARGS__) 4088e25f19bSMatthew Dillon 4098e25f19bSMatthew Dillon #define dfatal_errno(fmt, ...) \ 4108e25f19bSMatthew Dillon _dfatal(__FILE__, __LINE__, __func__, 1, fmt, ## __VA_ARGS__) 4118e25f19bSMatthew Dillon 4128e25f19bSMatthew Dillon #define dpanic_errno(fmt, ...) \ 4138e25f19bSMatthew Dillon _dfatal(__FILE__, __LINE__, __func__, 3, fmt, ## __VA_ARGS__) 4148e25f19bSMatthew Dillon 4158e25f19bSMatthew Dillon #define ddprintf(tab, fmt, ...) \ 41654f2fefcSMatthew Dillon do { if (DebugOpt >= 2) _ddprintf(tab, fmt, ## __VA_ARGS__); } while(0) 4178e25f19bSMatthew Dillon 4186fd67931SMatthew Dillon /* 4196fd67931SMatthew Dillon * addbuildenv() types 4206fd67931SMatthew Dillon */ 4216fd67931SMatthew Dillon #define BENV_ENVIRONMENT 1 4226fd67931SMatthew Dillon #define BENV_MAKECONF 2 42331f2ea22SMatthew Dillon #define BENV_CMDMASK 0x000F 42431f2ea22SMatthew Dillon 42531f2ea22SMatthew Dillon #define BENV_PKGLIST 0x0010 4266fd67931SMatthew Dillon 4276fd67931SMatthew Dillon /* 4286fd67931SMatthew Dillon * WORKER process flags 4296fd67931SMatthew Dillon */ 4306fd67931SMatthew Dillon #define WORKER_PROC_DEBUGSTOP 0x0001 4316fd67931SMatthew Dillon #define WORKER_PROC_DEVELOPER 0x0002 43232f62172SMatthew Dillon #define WORKER_PROC_CHECK_PLIST 0x0004 4336fd67931SMatthew Dillon 4346fd67931SMatthew Dillon /* 4356fd67931SMatthew Dillon * Misc 4366fd67931SMatthew Dillon */ 4378e25f19bSMatthew Dillon #define DOSTRING(label) #label 4388e25f19bSMatthew Dillon #define SCRIPTPATH(x) DOSTRING(x) 4396fd67931SMatthew Dillon #define MAXCAC 256 4408e25f19bSMatthew Dillon 441ea37671dSMatthew Dillon /* 442ea37671dSMatthew Dillon * RunStats satellite modules 443ea37671dSMatthew Dillon */ 444ea37671dSMatthew Dillon typedef struct topinfo { 4458b485838SMatthew Dillon int active; 446ea37671dSMatthew Dillon int pkgimpulse; 447ea37671dSMatthew Dillon int pkgrate; 448ea37671dSMatthew Dillon int noswap; 449ea37671dSMatthew Dillon int h; 450ea37671dSMatthew Dillon int m; 451ea37671dSMatthew Dillon int s; 452ea37671dSMatthew Dillon int total; 453ea37671dSMatthew Dillon int successful; 454ea37671dSMatthew Dillon int ignored; 455ea37671dSMatthew Dillon int remaining; 456ea37671dSMatthew Dillon int failed; 457ea37671dSMatthew Dillon int skipped; 458aac7a6d9SMatthew Dillon int dynmaxworkers; 459ea37671dSMatthew Dillon double dswap; 460ea37671dSMatthew Dillon double dload[3]; 461ea37671dSMatthew Dillon } topinfo_t; 462ea37671dSMatthew Dillon 463ea37671dSMatthew Dillon typedef struct runstats { 464ea37671dSMatthew Dillon struct runstats *next; 465ea37671dSMatthew Dillon void (*init)(void); 466ea37671dSMatthew Dillon void (*done)(void); 467ea37671dSMatthew Dillon void (*reset)(void); 468aac7a6d9SMatthew Dillon void (*update)(worker_t *work, const char *portdir); 469ea37671dSMatthew Dillon void (*updateTop)(topinfo_t *info); 470ea37671dSMatthew Dillon void (*updateLogs)(void); 4718b485838SMatthew Dillon void (*updateCompletion)(worker_t *work, int dlogid, 4728b485838SMatthew Dillon pkg_t *pkg, const char *reason); 473ea37671dSMatthew Dillon void (*sync)(void); 474ea37671dSMatthew Dillon } runstats_t; 475ea37671dSMatthew Dillon 476aac7a6d9SMatthew Dillon typedef struct monitorlog { 477aac7a6d9SMatthew Dillon off_t offset; 478aac7a6d9SMatthew Dillon int fd; 479aac7a6d9SMatthew Dillon int buf_beg; 480aac7a6d9SMatthew Dillon int buf_end; 481aac7a6d9SMatthew Dillon int buf_scan; 482aac7a6d9SMatthew Dillon int buf_discard_mode; 483aac7a6d9SMatthew Dillon char buf[1024]; 484aac7a6d9SMatthew Dillon } monitorlog_t; 485aac7a6d9SMatthew Dillon 486ea37671dSMatthew Dillon extern runstats_t NCursesRunStats; 487ea37671dSMatthew Dillon extern runstats_t MonitorRunStats; 488ea37671dSMatthew Dillon extern runstats_t HtmlRunStats; 489ea37671dSMatthew Dillon 4908e25f19bSMatthew Dillon extern int BuildCount; 4918e25f19bSMatthew Dillon extern int BuildTotal; 4928e25f19bSMatthew Dillon extern int BuildFailCount; 4938e25f19bSMatthew Dillon extern int BuildSkipCount; 49487017ac4SMatthew Dillon extern int BuildIgnoreCount; 4958e25f19bSMatthew Dillon extern int BuildSuccessCount; 496f7f25838SMatthew Dillon extern int DynamicMaxWorkers; 4978e25f19bSMatthew Dillon 4988e25f19bSMatthew Dillon extern buildenv_t *BuildEnv; 4996fd67931SMatthew Dillon extern int WorkerProcFlags; 5008e25f19bSMatthew Dillon extern int DebugOpt; 501927e4e3eSMatthew Dillon extern int MaskProbeAbort; 502a574cbf0SMatthew Dillon extern int ColorOpt; 503f7f25838SMatthew Dillon extern int SlowStartOpt; 5041645cafeSMatthew Dillon extern int YesOpt; 5059c4c701fSMatthew Dillon extern int NullStdinOpt; 506dc46751bSMatthew Dillon extern int DeleteObsoletePkgs; 5078e25f19bSMatthew Dillon extern int UseCCache; 508f4094b20SMatthew Dillon extern int UseUsrSrc; 5098e25f19bSMatthew Dillon extern int UseTmpfs; 5108e25f19bSMatthew Dillon extern int NumCores; 5117f0eca56SMatthew Dillon extern long PhysMem; 5127f0eca56SMatthew Dillon extern long PkgDepMemoryTarget; 5138e25f19bSMatthew Dillon extern int MaxBulk; 5148e25f19bSMatthew Dillon extern int MaxWorkers; 5158e25f19bSMatthew Dillon extern int MaxJobs; 5168e25f19bSMatthew Dillon extern int UseTmpfsWork; 5178e25f19bSMatthew Dillon extern int UseTmpfsBase; 5188e25f19bSMatthew Dillon extern int UseNCurses; 5198e25f19bSMatthew Dillon extern int LeveragePrebuilt; 5208e25f19bSMatthew Dillon extern char *DSynthExecPath; 5219e1d0b12SMatthew Dillon extern char *ProfileOverrideOpt; 522a67bf8dbSMatthew Dillon 5238e25f19bSMatthew Dillon extern const char *OperatingSystemName; 5248e25f19bSMatthew Dillon extern const char *ArchitectureName; 5258e25f19bSMatthew Dillon extern const char *MachineName; 5268e25f19bSMatthew Dillon extern const char *ReleaseName; 5278e25f19bSMatthew Dillon extern const char *VersionName; 5280df5ec4eSMatthew Dillon extern const char *VersionOnlyName; 5290df5ec4eSMatthew Dillon extern const char *VersionFromParamHeader; 5308e25f19bSMatthew Dillon 53168dc2eeaSMatthew Dillon extern const char *ConfigBase1; 53268dc2eeaSMatthew Dillon extern const char *ConfigBase2; 5338ec23ca1SMatthew Dillon extern const char *ConfigBase; 5348e25f19bSMatthew Dillon extern const char *DPortsPath; 5358e25f19bSMatthew Dillon extern const char *CCachePath; 5368e25f19bSMatthew Dillon extern const char *PackagesPath; 5378e25f19bSMatthew Dillon extern const char *RepositoryPath; 5388e25f19bSMatthew Dillon extern const char *OptionsPath; 5398e25f19bSMatthew Dillon extern const char *DistFilesPath; 5408e25f19bSMatthew Dillon extern const char *BuildBase; 5418e25f19bSMatthew Dillon extern const char *LogsPath; 5428e25f19bSMatthew Dillon extern const char *SystemPath; 543483dbac9SMatthew Dillon extern const char *UsePkgSufx; 54468dc2eeaSMatthew Dillon extern const char *Profile; 545aac7a6d9SMatthew Dillon extern char *StatsBase; 546aac7a6d9SMatthew Dillon extern char *StatsFilePath; 547aac7a6d9SMatthew Dillon extern char *StatsLockPath; 54868dc2eeaSMatthew Dillon 54968dc2eeaSMatthew Dillon extern int UsingHooks; 55068dc2eeaSMatthew Dillon extern const char *HookRunStart; 55168dc2eeaSMatthew Dillon extern const char *HookRunEnd; 55268dc2eeaSMatthew Dillon extern const char *HookPkgSuccess; 55368dc2eeaSMatthew Dillon extern const char *HookPkgFailure; 55468dc2eeaSMatthew Dillon extern const char *HookPkgIgnored; 55568dc2eeaSMatthew Dillon extern const char *HookPkgSkipped; 5568e25f19bSMatthew Dillon 5578e25f19bSMatthew Dillon void _dfatal(const char *file, int line, const char *func, int do_errno, 5588e25f19bSMatthew Dillon const char *fmt, ...); 5598e25f19bSMatthew Dillon void _ddprintf(int tab, const char *fmt, ...); 5608e25f19bSMatthew Dillon void _dlog(int which, const char *fmt, ...); 5614ea2ee4dSMatthew Dillon char *strdup_or_null(char *str); 5628e25f19bSMatthew Dillon void dlogreset(void); 56354f2fefcSMatthew Dillon int dlog00_fd(void); 5646fd67931SMatthew Dillon void addbuildenv(const char *label, const char *data, int type); 565a67bf8dbSMatthew Dillon void delbuildenv(const char *label); 566aac7a6d9SMatthew Dillon int readlogline(monitorlog_t *log, char **bufp); 5678e25f19bSMatthew Dillon 5688e25f19bSMatthew Dillon void initbulk(void (*func)(bulk_t *bulk), int jobs); 5697f0eca56SMatthew Dillon void queuebulk(const char *s1, const char *s2, const char *s3, 5708e25f19bSMatthew Dillon const char *s4); 5718e25f19bSMatthew Dillon bulk_t *getbulk(void); 5728e25f19bSMatthew Dillon void donebulk(void); 5738e25f19bSMatthew Dillon void freebulk(bulk_t *bulk); 5748e25f19bSMatthew Dillon void freestrp(char **strp); 5758e25f19bSMatthew Dillon void dupstrp(char **strp); 5761645cafeSMatthew Dillon int askyn(const char *ctl, ...); 5773699ee09SMatthew Dillon double getswappct(int *noswapp); 57868dc2eeaSMatthew Dillon FILE *dexec_open(const char **cav, int cac, pid_t *pidp, buildenv_t *xenv, 5796fd67931SMatthew Dillon int with_env, int with_mvars); 5806fd67931SMatthew Dillon int dexec_close(FILE *fp, pid_t pid); 5813cebe4a8SMatthew Dillon const char *getphasestr(worker_phase_t phase); 5828e25f19bSMatthew Dillon 5838e25f19bSMatthew Dillon void ParseConfiguration(int isworker); 584710838f7SMatthew Dillon pkg_t *ParsePackageList(int ac, char **av, int debugstop); 5858e25f19bSMatthew Dillon void FreePackageList(pkg_t *pkgs); 5868e25f19bSMatthew Dillon pkg_t *GetLocalPackageList(void); 5878e25f19bSMatthew Dillon pkg_t *GetFullPackageList(void); 5888e25f19bSMatthew Dillon pkg_t *GetPkgPkg(pkg_t *list); 5898e25f19bSMatthew Dillon 5908e25f19bSMatthew Dillon void DoConfigure(void); 5911645cafeSMatthew Dillon void DoStatus(pkg_t *pkgs); 5928e25f19bSMatthew Dillon void DoBuild(pkg_t *pkgs); 5938e25f19bSMatthew Dillon void DoInitBuild(int slot_override); 59487017ac4SMatthew Dillon void DoCleanBuild(int resetlogs); 595a67bf8dbSMatthew Dillon void OptimizeEnv(void); 5968e25f19bSMatthew Dillon void WorkerProcess(int ac, char **av); 5978e25f19bSMatthew Dillon 5981d6e00cdSMatthew Dillon int DoCreateTemplate(int force); 5998e25f19bSMatthew Dillon void DoDestroyTemplate(void); 6008e25f19bSMatthew Dillon void DoWorkerMounts(worker_t *work); 6018e25f19bSMatthew Dillon void DoWorkerUnmounts(worker_t *work); 6028e25f19bSMatthew Dillon void DoRebuildRepo(int ask); 6038e25f19bSMatthew Dillon void DoUpgradePkgs(pkg_t *pkgs, int ask); 6048e25f19bSMatthew Dillon void RemovePackages(pkg_t *pkgs); 6058e25f19bSMatthew Dillon void PurgeDistfiles(pkg_t *pkgs); 6068e25f19bSMatthew Dillon 607ea37671dSMatthew Dillon void RunStatsInit(void); 608ea37671dSMatthew Dillon void RunStatsDone(void); 609ea37671dSMatthew Dillon void RunStatsReset(void); 610aac7a6d9SMatthew Dillon void RunStatsUpdate(worker_t *work, const char *portdir); 6118b485838SMatthew Dillon void RunStatsUpdateTop(int active); 612ea37671dSMatthew Dillon void RunStatsUpdateLogs(void); 613ea37671dSMatthew Dillon void RunStatsSync(void); 6148b485838SMatthew Dillon void RunStatsUpdateCompletion(worker_t *work, int logid, 6158b485838SMatthew Dillon pkg_t *pkg, const char *reason); 6168e25f19bSMatthew Dillon 6178b485838SMatthew Dillon int copyfile(char *src, char *dst); 6188e25f19bSMatthew Dillon int ipcreadmsg(int fd, wmsg_t *msg); 6198e25f19bSMatthew Dillon int ipcwritemsg(int fd, wmsg_t *msg); 620aac7a6d9SMatthew Dillon extern void MonitorDirective(const char *datfile, const char *lkfile); 621