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> 458e25f19bSMatthew Dillon #include <stdio.h> 468e25f19bSMatthew Dillon #include <stdlib.h> 478e25f19bSMatthew Dillon #include <stddef.h> 488e25f19bSMatthew Dillon #include <stdarg.h> 498e25f19bSMatthew Dillon #include <unistd.h> 508e25f19bSMatthew Dillon #include <string.h> 518e25f19bSMatthew Dillon #include <fcntl.h> 528e25f19bSMatthew Dillon #include <signal.h> 538e25f19bSMatthew Dillon #include <poll.h> 548e25f19bSMatthew Dillon #include <assert.h> 558e25f19bSMatthew Dillon #include <errno.h> 568e25f19bSMatthew Dillon #include <pthread.h> 578e25f19bSMatthew Dillon #include <dirent.h> 588e25f19bSMatthew Dillon #include <termios.h> 598e25f19bSMatthew Dillon #include <ctype.h> 608e25f19bSMatthew Dillon 6168dc2eeaSMatthew Dillon /* 6268dc2eeaSMatthew Dillon * More esoteric headers 6368dc2eeaSMatthew Dillon */ 6468dc2eeaSMatthew Dillon #include <libutil.h> /* forkpty() */ 6568dc2eeaSMatthew Dillon #include <arpa/inet.h> /* ntohl() */ 6668dc2eeaSMatthew Dillon #include <elf.h> /* try to get elf info */ 67a574cbf0SMatthew Dillon 688e25f19bSMatthew Dillon struct pkglink; 698e25f19bSMatthew Dillon 708e25f19bSMatthew Dillon #define DSYNTH_VERSION "1.01" 718e25f19bSMatthew Dillon #define MAXWORKERS 1024 72*aac7a6d9SMatthew Dillon #define MAXLOGLINES 1024 738e25f19bSMatthew Dillon #define MAXJOBS 8192 /* just used for -j sanity */ 748e25f19bSMatthew Dillon #define MAXBULK MAXWORKERS 758e25f19bSMatthew Dillon 768e25f19bSMatthew Dillon #define MAKE_BINARY "/usr/bin/make" 778e25f19bSMatthew Dillon #define PKG_BINARY "/usr/local/sbin/pkg" 788e25f19bSMatthew Dillon #define MOUNT_BINARY "/sbin/mount" 798e25f19bSMatthew Dillon #define MOUNT_NULLFS_BINARY "/sbin/mount_null" 808e25f19bSMatthew Dillon #define MOUNT_TMPFS_BINARY "/sbin/mount_tmpfs" 818e25f19bSMatthew Dillon #define MOUNT_DEVFS_BINARY "/sbin/mount_devfs" 828ec23ca1SMatthew Dillon #define MOUNT_PROCFS_BINARY "/sbin/mount_procfs" 838e25f19bSMatthew Dillon #define UMOUNT_BINARY "/sbin/umount" 848e25f19bSMatthew Dillon 85*aac7a6d9SMatthew Dillon #define STATS_FILE "monitor.dat" /* under LogsPath */ 86*aac7a6d9SMatthew Dillon #define STATS_LOCKFILE "monitor.lk" /* under LogsPath */ 87*aac7a6d9SMatthew Dillon 887f0eca56SMatthew Dillon #define ONEGB (1024L * 1024 * 1024) 89f4094b20SMatthew Dillon #define DISABLED_STR "disabled" 907f0eca56SMatthew Dillon 918e25f19bSMatthew Dillon /* 92667fb2cbSMatthew Dillon * This can be ".tar", ".tgz", ".txz", or ".tbz". 93667fb2cbSMatthew Dillon * 94667fb2cbSMatthew Dillon * .tar - very fast but you'll need 1TB+ of storage just for the package files. 95667fb2cbSMatthew Dillon * .txz - very compact but decompression speed is horrible. 96667fb2cbSMatthew Dillon * .tgz - reasonable compression, extremely fast decompression. Roughly 97667fb2cbSMatthew Dillon * 1.1x to 2.0x the size of a .txz, but decompresses 10x faster. 98667fb2cbSMatthew Dillon * .tbz - worse than .tgz generally 99667fb2cbSMatthew Dillon */ 100667fb2cbSMatthew Dillon #define USE_PKG_SUFX ".tgz" 101667fb2cbSMatthew Dillon 102667fb2cbSMatthew Dillon /* 1038e25f19bSMatthew Dillon * Topology linkages 1048e25f19bSMatthew Dillon */ 1058e25f19bSMatthew Dillon typedef struct pkglink { 1068e25f19bSMatthew Dillon struct pkglink *next; 1078e25f19bSMatthew Dillon struct pkglink *prev; 1088e25f19bSMatthew Dillon struct pkg *pkg; 10963fcce5bSMatthew Dillon int dep_type; 1108e25f19bSMatthew Dillon } pkglink_t; 1118e25f19bSMatthew Dillon 11263fcce5bSMatthew Dillon #define DEP_TYPE_FETCH 1 11363fcce5bSMatthew Dillon #define DEP_TYPE_EXT 2 11463fcce5bSMatthew Dillon #define DEP_TYPE_PATCH 3 11563fcce5bSMatthew Dillon #define DEP_TYPE_BUILD 4 11663fcce5bSMatthew Dillon #define DEP_TYPE_LIB 5 11763fcce5bSMatthew Dillon #define DEP_TYPE_RUN 6 11863fcce5bSMatthew Dillon 1198e25f19bSMatthew Dillon /* 1208e25f19bSMatthew Dillon * Describes a [flavored] package 1218e25f19bSMatthew Dillon */ 1228e25f19bSMatthew Dillon typedef struct pkg { 1238e25f19bSMatthew Dillon struct pkg *build_next; /* topology inversion build list */ 1248e25f19bSMatthew Dillon struct pkg *bnext; /* linked list from bulk return */ 1258e25f19bSMatthew Dillon struct pkg *hnext1; /* hash based on portdir */ 1268e25f19bSMatthew Dillon struct pkg *hnext2; /* hash based on pkgfile */ 1278e25f19bSMatthew Dillon pkglink_t idepon_list; /* I need these pkgs */ 1288e25f19bSMatthew Dillon pkglink_t deponi_list; /* pkgs which depend on me */ 1298e25f19bSMatthew Dillon char *portdir; /* origin name e.g. www/chromium[@flavor] */ 1308e25f19bSMatthew Dillon char *logfile; /* relative logfile path */ 1318e25f19bSMatthew Dillon char *version; /* PKGVERSION - e.g. 3.5.0_1 */ 1328e25f19bSMatthew Dillon char *pkgfile; /* PKGFILE - e.g. flav-blah-3.5.0_1.txz */ 1331645cafeSMatthew Dillon char *distfiles; /* DISTFILES - e.g. blah-68.0.source.tar.xz */ 1341645cafeSMatthew Dillon char *distsubdir; /* DIST_SUBDIR- e.g. cabal */ 13587017ac4SMatthew Dillon char *ignore; /* IGNORE (also covers BROKEN) */ 1368e25f19bSMatthew Dillon char *fetch_deps; /* FETCH_DEPENDS */ 1378e25f19bSMatthew Dillon char *ext_deps; /* EXTRACT_DEPENDS */ 1388e25f19bSMatthew Dillon char *patch_deps; /* PATCH_DEPENDS */ 1398e25f19bSMatthew Dillon char *build_deps; /* BUILD_DEPENDS */ 1408e25f19bSMatthew Dillon char *lib_deps; /* LIB_DEPENDS */ 1418e25f19bSMatthew Dillon char *run_deps; /* RUN_DEPENDS */ 1428e25f19bSMatthew Dillon char *pos_options; /* SELECTED_OPTIONS */ 1438e25f19bSMatthew Dillon char *neg_options; /* DESELECTED_OPTIONS */ 1448e25f19bSMatthew Dillon char *flavors; /* FLAVORS - e.g. py36 py27 */ 145f4094b20SMatthew Dillon char *uses; /* USES (metaport test) */ 1468e25f19bSMatthew Dillon int make_jobs_number; /* MAKE_JOBS_NUMBER */ 1478e25f19bSMatthew Dillon int use_linux; /* USE_LINUX */ 1488e25f19bSMatthew Dillon int idep_count; /* count recursive idepon build deps */ 1498e25f19bSMatthew Dillon int depi_count; /* count recursive deponi build deps */ 15088c24d72SMatthew Dillon int depi_depth; /* tree depth who depends on me */ 1518e25f19bSMatthew Dillon int dsynth_install_flg; /* locked with WorkerMutex */ 1528e25f19bSMatthew Dillon int flags; 153fef2fc63SMatthew Dillon size_t pkgfile_size; /* size of pkgfile */ 1548e25f19bSMatthew Dillon } pkg_t; 1558e25f19bSMatthew Dillon 1568e25f19bSMatthew Dillon #define PKGF_PACKAGED 0x00000001 /* has a repo package */ 1578e25f19bSMatthew Dillon #define PKGF_DUMMY 0x00000002 /* generic root for flavors */ 1588e25f19bSMatthew Dillon #define PKGF_NOTFOUND 0x00000004 /* dport not found */ 1598e25f19bSMatthew Dillon #define PKGF_CORRUPT 0x00000008 /* dport corrupt */ 1608e25f19bSMatthew Dillon #define PKGF_PLACEHOLD 0x00000010 /* pre-entered */ 1618e25f19bSMatthew Dillon #define PKGF_BUILDLIST 0x00000020 /* on build_list */ 1628e25f19bSMatthew Dillon #define PKGF_BUILDLOOP 0x00000040 /* traversal loop test */ 1638e25f19bSMatthew Dillon #define PKGF_BUILDTRAV 0x00000080 /* traversal optimization */ 1648e25f19bSMatthew Dillon #define PKGF_NOBUILD_D 0x00000100 /* can't build - dependency problem */ 1658e25f19bSMatthew Dillon #define PKGF_NOBUILD_S 0x00000200 /* can't build - skipped */ 1668e25f19bSMatthew Dillon #define PKGF_NOBUILD_F 0x00000400 /* can't build - failed */ 16787017ac4SMatthew Dillon #define PKGF_NOBUILD_I 0x00000800 /* can't build - ignored or broken */ 16887017ac4SMatthew Dillon #define PKGF_SUCCESS 0x00001000 /* build complete */ 16987017ac4SMatthew Dillon #define PKGF_FAILURE 0x00002000 /* build complete */ 17087017ac4SMatthew Dillon #define PKGF_RUNNING 0x00004000 /* build complete */ 17187017ac4SMatthew Dillon #define PKGF_PKGPKG 0x00008000 /* pkg/pkg-static special */ 17287017ac4SMatthew Dillon #define PKGF_NOTREADY 0x00010000 /* build_find_leaves() only */ 173f4094b20SMatthew Dillon #define PKGF_MANUALSEL 0x00020000 /* manually specified */ 174f4094b20SMatthew Dillon #define PKGF_META 0x00040000 /* USES contains 'metaport' */ 1758e25f19bSMatthew Dillon #define PKGF_ERROR (PKGF_PLACEHOLD | PKGF_CORRUPT | PKGF_NOTFOUND | \ 1768e25f19bSMatthew Dillon PKGF_FAILURE) 17787017ac4SMatthew Dillon #define PKGF_NOBUILD (PKGF_NOBUILD_D | PKGF_NOBUILD_S | PKGF_NOBUILD_F | \ 17887017ac4SMatthew Dillon PKGF_NOBUILD_I) 1798e25f19bSMatthew Dillon 1808e25f19bSMatthew Dillon #define PKGLIST_EMPTY(pkglink) ((pkglink)->next == (pkglink)) 1818e25f19bSMatthew Dillon #define PKGLIST_FOREACH(var, head) \ 1828e25f19bSMatthew Dillon for (var = (head)->next; var != (head); var = (var)->next) 1838e25f19bSMatthew Dillon 1848e25f19bSMatthew Dillon typedef struct bulk { 1858e25f19bSMatthew Dillon struct bulk *next; 1868e25f19bSMatthew Dillon pthread_t td; 1878e25f19bSMatthew Dillon int debug; 1888e25f19bSMatthew Dillon int flags; 1898e25f19bSMatthew Dillon enum { UNLISTED, ONSUBMIT, ONRUN, ISRUNNING, ONRESPONSE } state; 1908e25f19bSMatthew Dillon char *s1; 1918e25f19bSMatthew Dillon char *s2; 1928e25f19bSMatthew Dillon char *s3; 1938e25f19bSMatthew Dillon char *s4; 1948e25f19bSMatthew Dillon char *r1; 1958e25f19bSMatthew Dillon char *r2; 1968e25f19bSMatthew Dillon char *r3; 1978e25f19bSMatthew Dillon char *r4; 1988e25f19bSMatthew Dillon pkg_t *list; /* pkgs linked by bnext */ 1998e25f19bSMatthew Dillon } bulk_t; 2008e25f19bSMatthew Dillon 2018e25f19bSMatthew Dillon /* 2028e25f19bSMatthew Dillon * Worker state (up to MAXWORKERS). Each worker operates within a 2038e25f19bSMatthew Dillon * chroot or jail. A system mirror is setup and the template 2048e25f19bSMatthew Dillon * is copied in. 2058e25f19bSMatthew Dillon * 2068e25f19bSMatthew Dillon * basedir - tmpfs 2078e25f19bSMatthew Dillon * /bin - nullfs (ro) 2088e25f19bSMatthew Dillon * /sbin - nullfs (ro) 2098e25f19bSMatthew Dillon * /lib - nullfs (ro) 2108e25f19bSMatthew Dillon * /libexec - nullfs (ro) 2118e25f19bSMatthew Dillon * /usr/bin - nullfs (ro) 2128e25f19bSMatthew Dillon * /usr/include - nullfs (ro) 2138e25f19bSMatthew Dillon * /usr/lib - nullfs (ro) 2148e25f19bSMatthew Dillon * /usr/libdata - nullfs (ro) 2158e25f19bSMatthew Dillon * /usr/libexec - nullfs (ro) 2168e25f19bSMatthew Dillon * /usr/sbin - nullfs (ro) 2178e25f19bSMatthew Dillon * /usr/share - nullfs (ro) 2188e25f19bSMatthew Dillon * /xports - nullfs (ro) 2198e25f19bSMatthew Dillon * /options - nullfs (ro) 2208e25f19bSMatthew Dillon * /packages - nullfs (ro) 2218e25f19bSMatthew Dillon * /distfiles - nullfs (ro) 2228e25f19bSMatthew Dillon * construction - tmpfs 2238e25f19bSMatthew Dillon * /usr/local - tmpfs 2248e25f19bSMatthew Dillon * /boot - nullfs (ro) 2258e25f19bSMatthew Dillon * /boot/modules.local - tmpfs 2268e25f19bSMatthew Dillon * /usr/games - nullfs (ro) 2278e25f19bSMatthew Dillon * /usr/src - nullfs (ro) 2288e25f19bSMatthew Dillon * /dev - devfs 2298e25f19bSMatthew Dillon */ 2308e25f19bSMatthew Dillon enum worker_state { WORKER_NONE, WORKER_IDLE, WORKER_PENDING, 2318e25f19bSMatthew Dillon WORKER_RUNNING, WORKER_DONE, WORKER_FAILED, 2328ec23ca1SMatthew Dillon WORKER_FROZEN, WORKER_EXITING }; 2338e25f19bSMatthew Dillon typedef enum worker_state worker_state_t; 2348e25f19bSMatthew Dillon 2358e25f19bSMatthew Dillon enum worker_phase { PHASE_PENDING, 2368e25f19bSMatthew Dillon PHASE_INSTALL_PKGS, 2378e25f19bSMatthew Dillon PHASE_CHECK_SANITY, 2388e25f19bSMatthew Dillon PHASE_PKG_DEPENDS, 2398e25f19bSMatthew Dillon PHASE_FETCH_DEPENDS, 2408e25f19bSMatthew Dillon PHASE_FETCH, 2418e25f19bSMatthew Dillon PHASE_CHECKSUM, 2428e25f19bSMatthew Dillon PHASE_EXTRACT_DEPENDS, 2438e25f19bSMatthew Dillon PHASE_EXTRACT, 2448e25f19bSMatthew Dillon PHASE_PATCH_DEPENDS, 2458e25f19bSMatthew Dillon PHASE_PATCH, 2468e25f19bSMatthew Dillon PHASE_BUILD_DEPENDS, 2478e25f19bSMatthew Dillon PHASE_LIB_DEPENDS, 2488e25f19bSMatthew Dillon PHASE_CONFIGURE, 2498e25f19bSMatthew Dillon PHASE_BUILD, 2508e25f19bSMatthew Dillon PHASE_RUN_DEPENDS, 2518e25f19bSMatthew Dillon PHASE_STAGE, 2528e25f19bSMatthew Dillon PHASE_TEST, 2538e25f19bSMatthew Dillon PHASE_CHECK_PLIST, 2548e25f19bSMatthew Dillon PHASE_PACKAGE, 2558e25f19bSMatthew Dillon PHASE_INSTALL_MTREE, 2568e25f19bSMatthew Dillon PHASE_INSTALL, 2578e25f19bSMatthew Dillon PHASE_DEINSTALL 2588e25f19bSMatthew Dillon }; 2598e25f19bSMatthew Dillon 2608e25f19bSMatthew Dillon typedef enum worker_phase worker_phase_t; 2618e25f19bSMatthew Dillon 2628e25f19bSMatthew Dillon /* 2638e25f19bSMatthew Dillon * Watchdog timeouts, in minutes, baseline, scales up with load/ncpus but 2648e25f19bSMatthew Dillon * does not scale down. 2658e25f19bSMatthew Dillon */ 2668e25f19bSMatthew Dillon #define WDOG1 (5) 2678e25f19bSMatthew Dillon #define WDOG2 (10) 2688e25f19bSMatthew Dillon #define WDOG3 (15) 2698e25f19bSMatthew Dillon #define WDOG4 (30) 2708e25f19bSMatthew Dillon #define WDOG5 (60) 2718e25f19bSMatthew Dillon #define WDOG6 (60 + 30) 2728e25f19bSMatthew Dillon #define WDOG7 (60 * 2) 2738e25f19bSMatthew Dillon #define WDOG8 (60 * 2 + 30) 2748e25f19bSMatthew Dillon #define WDOG9 (60 * 3) 2758e25f19bSMatthew Dillon 2768e25f19bSMatthew Dillon typedef struct worker { 2778e25f19bSMatthew Dillon int index; /* worker number 0..N-1 */ 2788e25f19bSMatthew Dillon int flags; 2798e25f19bSMatthew Dillon int accum_error; /* cumulative error */ 2808e25f19bSMatthew Dillon int mount_error; /* mount and unmount error */ 2818e25f19bSMatthew Dillon int terminate : 1; /* request sub-thread to terminate */ 2828e25f19bSMatthew Dillon char *basedir; /* base directory including id */ 2838e25f19bSMatthew Dillon char *flavor; 2848e25f19bSMatthew Dillon pthread_t td; /* pthread */ 2858e25f19bSMatthew Dillon pthread_cond_t cond; /* interlock cond (w/ WorkerMutex) */ 2868e25f19bSMatthew Dillon pkg_t *pkg; 2878e25f19bSMatthew Dillon worker_state_t state; /* general worker state */ 2888e25f19bSMatthew Dillon worker_phase_t phase; /* phase control in childBuilderThread */ 2898e25f19bSMatthew Dillon time_t start_time; 2908e25f19bSMatthew Dillon long lines; 291516819d9SMatthew Dillon long memuse; 2928e25f19bSMatthew Dillon pid_t pid; 2938e25f19bSMatthew Dillon int fds[2]; /* forked environment process */ 2948e25f19bSMatthew Dillon char status[64]; 295fef2fc63SMatthew Dillon size_t pkg_dep_size; /* pkg dependency size(s) */ 2968e25f19bSMatthew Dillon } worker_t; 2978e25f19bSMatthew Dillon 2988e25f19bSMatthew Dillon #define WORKERF_STATUS_UPDATE 0x0001 /* display update */ 2998e25f19bSMatthew Dillon #define WORKERF_SUCCESS 0x0002 /* completion flag */ 3008e25f19bSMatthew Dillon #define WORKERF_FAILURE 0x0004 /* completion flag */ 3018ec23ca1SMatthew Dillon #define WORKERF_FREEZE 0x0008 /* freeze the worker */ 3028e25f19bSMatthew Dillon 3038e25f19bSMatthew Dillon #define MOUNT_TYPE_MASK 0x000F 3048e25f19bSMatthew Dillon #define MOUNT_TYPE_TMPFS 0x0001 3058e25f19bSMatthew Dillon #define MOUNT_TYPE_NULLFS 0x0002 3068e25f19bSMatthew Dillon #define MOUNT_TYPE_DEVFS 0x0003 3078ec23ca1SMatthew Dillon #define MOUNT_TYPE_PROCFS 0x0004 3088e25f19bSMatthew Dillon #define MOUNT_TYPE_RW 0x0010 3098e25f19bSMatthew Dillon #define MOUNT_TYPE_BIG 0x0020 3108e25f19bSMatthew Dillon #define MOUNT_TYPE_TMP 0x0040 3118e25f19bSMatthew Dillon 3128e25f19bSMatthew Dillon #define NULLFS_RO (MOUNT_TYPE_NULLFS) 3138e25f19bSMatthew Dillon #define NULLFS_RW (MOUNT_TYPE_NULLFS | MOUNT_TYPE_RW) 3148ec23ca1SMatthew Dillon #define PROCFS_RO (MOUNT_TYPE_PROCFS) 3158e25f19bSMatthew Dillon #define TMPFS_RW (MOUNT_TYPE_TMPFS | MOUNT_TYPE_RW) 3168e25f19bSMatthew Dillon #define TMPFS_RW_BIG (MOUNT_TYPE_TMPFS | MOUNT_TYPE_RW | \ 3178e25f19bSMatthew Dillon MOUNT_TYPE_BIG) 3188e25f19bSMatthew Dillon #define DEVFS_RW (MOUNT_TYPE_DEVFS | MOUNT_TYPE_RW) 3198e25f19bSMatthew Dillon 3208e25f19bSMatthew Dillon /* 3218e25f19bSMatthew Dillon * IPC messages between the worker support thread and the worker process. 3228e25f19bSMatthew Dillon */ 3238e25f19bSMatthew Dillon typedef struct wmsg { 3248e25f19bSMatthew Dillon int cmd; 3258e25f19bSMatthew Dillon int status; 3268e25f19bSMatthew Dillon long lines; 327516819d9SMatthew Dillon long memuse; 3288e25f19bSMatthew Dillon worker_phase_t phase; 3298e25f19bSMatthew Dillon } wmsg_t; 3308e25f19bSMatthew Dillon 3318e25f19bSMatthew Dillon #define WMSG_CMD_STATUS_UPDATE 0x0001 3328e25f19bSMatthew Dillon #define WMSG_CMD_SUCCESS 0x0002 3338e25f19bSMatthew Dillon #define WMSG_CMD_FAILURE 0x0003 3348e25f19bSMatthew Dillon #define WMSG_CMD_INSTALL_PKGS 0x0004 3358e25f19bSMatthew Dillon #define WMSG_RES_INSTALL_PKGS 0x0005 3368ec23ca1SMatthew Dillon #define WMSG_CMD_FREEZEWORKER 0x0006 3378e25f19bSMatthew Dillon 3388e25f19bSMatthew Dillon /* 3398e25f19bSMatthew Dillon * Make variables and build environment 3408e25f19bSMatthew Dillon */ 3418e25f19bSMatthew Dillon typedef struct buildenv { 3428e25f19bSMatthew Dillon struct buildenv *next; 34368dc2eeaSMatthew Dillon const char *label; 34468dc2eeaSMatthew Dillon const char *data; 34568dc2eeaSMatthew Dillon char *a1; /* allocations */ 34668dc2eeaSMatthew Dillon char *a2; /* allocations */ 3476fd67931SMatthew Dillon int type; 3488e25f19bSMatthew Dillon } buildenv_t; 3498e25f19bSMatthew Dillon 3508e25f19bSMatthew Dillon /* 3518e25f19bSMatthew Dillon * Operating systems recognized by dsynth 3528e25f19bSMatthew Dillon */ 3538e25f19bSMatthew Dillon enum os_id { 3548e25f19bSMatthew Dillon OS_UNKNOWN, OS_DRAGONFLY, OS_FREEBSD, OS_NETBSD, OS_LINUX 3558e25f19bSMatthew Dillon }; 3568e25f19bSMatthew Dillon 3578e25f19bSMatthew Dillon typedef enum os_id os_id_t; 3588e25f19bSMatthew Dillon 3598e25f19bSMatthew Dillon /* 3608e25f19bSMatthew Dillon * DLOG 3618e25f19bSMatthew Dillon */ 3628e25f19bSMatthew Dillon #define DLOG_ALL 0 /* Usually stdout when curses disabled */ 3638e25f19bSMatthew Dillon #define DLOG_SUCC 1 /* success_list.log */ 3648e25f19bSMatthew Dillon #define DLOG_FAIL 2 /* failure_list.log */ 3658e25f19bSMatthew Dillon #define DLOG_IGN 3 /* ignored_list.log */ 3668e25f19bSMatthew Dillon #define DLOG_SKIP 4 /* skipped_list.log */ 3678e25f19bSMatthew Dillon #define DLOG_ABN 5 /* abnormal_command_output */ 3688e25f19bSMatthew Dillon #define DLOG_OBS 6 /* obsolete_packages.log */ 3698e25f19bSMatthew Dillon #define DLOG_COUNT 7 /* total number of DLOGs */ 370a574cbf0SMatthew Dillon #define DLOG_MASK 0x0FF 371a574cbf0SMatthew Dillon 372a574cbf0SMatthew Dillon #define DLOG_FILTER 0x100 /* Filter out of stdout in non-curses mode */ 373a574cbf0SMatthew Dillon #define DLOG_RED 0x200 /* Print in color */ 374a574cbf0SMatthew Dillon #define DLOG_GRN 0x400 /* Print in color */ 3758e25f19bSMatthew Dillon 3768e25f19bSMatthew Dillon #define dassert(exp, fmt, ...) \ 3778e25f19bSMatthew Dillon if (!(exp)) dpanic(fmt, ## __VA_ARGS__) 3788e25f19bSMatthew Dillon 3798e25f19bSMatthew Dillon #define ddassert(exp) \ 3808e25f19bSMatthew Dillon dassert((exp), "\"%s\" line %d", __FILE__, __LINE__) 3818e25f19bSMatthew Dillon 3828e25f19bSMatthew Dillon #define dassert_errno(exp, fmt, ...) \ 3839c4c701fSMatthew Dillon if (!(exp)) dpanic_errno(fmt, ## __VA_ARGS__) 3848e25f19bSMatthew Dillon 3858e25f19bSMatthew Dillon #define dlog(which, fmt, ...) \ 3868e25f19bSMatthew Dillon _dlog(which, fmt, ## __VA_ARGS__) 3878e25f19bSMatthew Dillon 388a574cbf0SMatthew Dillon #define dlog_tsnl(which, fmt, ...) \ 389a574cbf0SMatthew Dillon _dlog(which, fmt, ## __VA_ARGS__) 390a574cbf0SMatthew Dillon 3918e25f19bSMatthew Dillon #define dfatal(fmt, ...) \ 3928e25f19bSMatthew Dillon _dfatal(__FILE__, __LINE__, __func__, 0, fmt, ## __VA_ARGS__) 3938e25f19bSMatthew Dillon 3948e25f19bSMatthew Dillon #define dpanic(fmt, ...) \ 3958e25f19bSMatthew Dillon _dfatal(__FILE__, __LINE__, __func__, 2, fmt, ## __VA_ARGS__) 3968e25f19bSMatthew Dillon 3978e25f19bSMatthew Dillon #define dfatal_errno(fmt, ...) \ 3988e25f19bSMatthew Dillon _dfatal(__FILE__, __LINE__, __func__, 1, fmt, ## __VA_ARGS__) 3998e25f19bSMatthew Dillon 4008e25f19bSMatthew Dillon #define dpanic_errno(fmt, ...) \ 4018e25f19bSMatthew Dillon _dfatal(__FILE__, __LINE__, __func__, 3, fmt, ## __VA_ARGS__) 4028e25f19bSMatthew Dillon 4038e25f19bSMatthew Dillon #define ddprintf(tab, fmt, ...) \ 40454f2fefcSMatthew Dillon do { if (DebugOpt >= 2) _ddprintf(tab, fmt, ## __VA_ARGS__); } while(0) 4058e25f19bSMatthew Dillon 4066fd67931SMatthew Dillon /* 4076fd67931SMatthew Dillon * addbuildenv() types 4086fd67931SMatthew Dillon */ 4096fd67931SMatthew Dillon #define BENV_ENVIRONMENT 1 4106fd67931SMatthew Dillon #define BENV_MAKECONF 2 41131f2ea22SMatthew Dillon #define BENV_CMDMASK 0x000F 41231f2ea22SMatthew Dillon 41331f2ea22SMatthew Dillon #define BENV_PKGLIST 0x0010 4146fd67931SMatthew Dillon 4156fd67931SMatthew Dillon /* 4166fd67931SMatthew Dillon * WORKER process flags 4176fd67931SMatthew Dillon */ 4186fd67931SMatthew Dillon #define WORKER_PROC_DEBUGSTOP 0x0001 4196fd67931SMatthew Dillon #define WORKER_PROC_DEVELOPER 0x0002 4206fd67931SMatthew Dillon 4216fd67931SMatthew Dillon /* 4226fd67931SMatthew Dillon * Misc 4236fd67931SMatthew Dillon */ 4248e25f19bSMatthew Dillon #define DOSTRING(label) #label 4258e25f19bSMatthew Dillon #define SCRIPTPATH(x) DOSTRING(x) 4266fd67931SMatthew Dillon #define MAXCAC 256 4278e25f19bSMatthew Dillon 428ea37671dSMatthew Dillon /* 429ea37671dSMatthew Dillon * RunStats satellite modules 430ea37671dSMatthew Dillon */ 431ea37671dSMatthew Dillon typedef struct topinfo { 432ea37671dSMatthew Dillon int pkgimpulse; 433ea37671dSMatthew Dillon int pkgrate; 434ea37671dSMatthew Dillon int noswap; 435ea37671dSMatthew Dillon int h; 436ea37671dSMatthew Dillon int m; 437ea37671dSMatthew Dillon int s; 438ea37671dSMatthew Dillon int total; 439ea37671dSMatthew Dillon int successful; 440ea37671dSMatthew Dillon int ignored; 441ea37671dSMatthew Dillon int remaining; 442ea37671dSMatthew Dillon int failed; 443ea37671dSMatthew Dillon int skipped; 444*aac7a6d9SMatthew Dillon int dynmaxworkers; 445ea37671dSMatthew Dillon double dswap; 446ea37671dSMatthew Dillon double dload[3]; 447ea37671dSMatthew Dillon } topinfo_t; 448ea37671dSMatthew Dillon 449ea37671dSMatthew Dillon typedef struct runstats { 450ea37671dSMatthew Dillon struct runstats *next; 451ea37671dSMatthew Dillon void (*init)(void); 452ea37671dSMatthew Dillon void (*done)(void); 453ea37671dSMatthew Dillon void (*reset)(void); 454*aac7a6d9SMatthew Dillon void (*update)(worker_t *work, const char *portdir); 455ea37671dSMatthew Dillon void (*updateTop)(topinfo_t *info); 456ea37671dSMatthew Dillon void (*updateLogs)(void); 457ea37671dSMatthew Dillon void (*sync)(void); 458ea37671dSMatthew Dillon } runstats_t; 459ea37671dSMatthew Dillon 460*aac7a6d9SMatthew Dillon typedef struct monitorlog { 461*aac7a6d9SMatthew Dillon off_t offset; 462*aac7a6d9SMatthew Dillon int fd; 463*aac7a6d9SMatthew Dillon int buf_beg; 464*aac7a6d9SMatthew Dillon int buf_end; 465*aac7a6d9SMatthew Dillon int buf_scan; 466*aac7a6d9SMatthew Dillon int buf_discard_mode; 467*aac7a6d9SMatthew Dillon char buf[1024]; 468*aac7a6d9SMatthew Dillon } monitorlog_t; 469*aac7a6d9SMatthew Dillon 470ea37671dSMatthew Dillon extern runstats_t NCursesRunStats; 471ea37671dSMatthew Dillon extern runstats_t MonitorRunStats; 472ea37671dSMatthew Dillon extern runstats_t HtmlRunStats; 473ea37671dSMatthew Dillon 4748e25f19bSMatthew Dillon extern int BuildCount; 4758e25f19bSMatthew Dillon extern int BuildTotal; 4768e25f19bSMatthew Dillon extern int BuildFailCount; 4778e25f19bSMatthew Dillon extern int BuildSkipCount; 47887017ac4SMatthew Dillon extern int BuildIgnoreCount; 4798e25f19bSMatthew Dillon extern int BuildSuccessCount; 480f7f25838SMatthew Dillon extern int DynamicMaxWorkers; 4818e25f19bSMatthew Dillon 4828e25f19bSMatthew Dillon extern buildenv_t *BuildEnv; 4836fd67931SMatthew Dillon extern int WorkerProcFlags; 4848e25f19bSMatthew Dillon extern int DebugOpt; 485a574cbf0SMatthew Dillon extern int ColorOpt; 486f7f25838SMatthew Dillon extern int SlowStartOpt; 4871645cafeSMatthew Dillon extern int YesOpt; 4889c4c701fSMatthew Dillon extern int NullStdinOpt; 4898e25f19bSMatthew Dillon extern int UseCCache; 490f4094b20SMatthew Dillon extern int UseUsrSrc; 4918e25f19bSMatthew Dillon extern int UseTmpfs; 4928e25f19bSMatthew Dillon extern int NumCores; 4937f0eca56SMatthew Dillon extern long PhysMem; 4947f0eca56SMatthew Dillon extern long PkgDepMemoryTarget; 4958e25f19bSMatthew Dillon extern int MaxBulk; 4968e25f19bSMatthew Dillon extern int MaxWorkers; 4978e25f19bSMatthew Dillon extern int MaxJobs; 4988e25f19bSMatthew Dillon extern int UseTmpfsWork; 4998e25f19bSMatthew Dillon extern int UseTmpfsBase; 5008e25f19bSMatthew Dillon extern int UseNCurses; 5018e25f19bSMatthew Dillon extern int LeveragePrebuilt; 5028e25f19bSMatthew Dillon extern char *DSynthExecPath; 5038e25f19bSMatthew Dillon 504a67bf8dbSMatthew Dillon 5058e25f19bSMatthew Dillon extern const char *OperatingSystemName; 5068e25f19bSMatthew Dillon extern const char *ArchitectureName; 5078e25f19bSMatthew Dillon extern const char *MachineName; 5088e25f19bSMatthew Dillon extern const char *ReleaseName; 5098e25f19bSMatthew Dillon extern const char *VersionName; 5100df5ec4eSMatthew Dillon extern const char *VersionOnlyName; 5110df5ec4eSMatthew Dillon extern const char *VersionFromParamHeader; 5128e25f19bSMatthew Dillon 51368dc2eeaSMatthew Dillon extern const char *ConfigBase1; 51468dc2eeaSMatthew Dillon extern const char *ConfigBase2; 5158ec23ca1SMatthew Dillon extern const char *ConfigBase; 5168e25f19bSMatthew Dillon extern const char *DPortsPath; 5178e25f19bSMatthew Dillon extern const char *CCachePath; 5188e25f19bSMatthew Dillon extern const char *SynthConfig; 5198e25f19bSMatthew Dillon extern const char *PackagesPath; 5208e25f19bSMatthew Dillon extern const char *RepositoryPath; 5218e25f19bSMatthew Dillon extern const char *OptionsPath; 5228e25f19bSMatthew Dillon extern const char *DistFilesPath; 5238e25f19bSMatthew Dillon extern const char *BuildBase; 5248e25f19bSMatthew Dillon extern const char *LogsPath; 5258e25f19bSMatthew Dillon extern const char *SystemPath; 52668dc2eeaSMatthew Dillon extern const char *Profile; 527*aac7a6d9SMatthew Dillon extern char *StatsBase; 528*aac7a6d9SMatthew Dillon extern char *StatsFilePath; 529*aac7a6d9SMatthew Dillon extern char *StatsLockPath; 53068dc2eeaSMatthew Dillon 53168dc2eeaSMatthew Dillon extern int UsingHooks; 53268dc2eeaSMatthew Dillon extern const char *HookRunStart; 53368dc2eeaSMatthew Dillon extern const char *HookRunEnd; 53468dc2eeaSMatthew Dillon extern const char *HookPkgSuccess; 53568dc2eeaSMatthew Dillon extern const char *HookPkgFailure; 53668dc2eeaSMatthew Dillon extern const char *HookPkgIgnored; 53768dc2eeaSMatthew Dillon extern const char *HookPkgSkipped; 5388e25f19bSMatthew Dillon 5398e25f19bSMatthew Dillon void _dfatal(const char *file, int line, const char *func, int do_errno, 5408e25f19bSMatthew Dillon const char *fmt, ...); 5418e25f19bSMatthew Dillon void _ddprintf(int tab, const char *fmt, ...); 5428e25f19bSMatthew Dillon void _dlog(int which, const char *fmt, ...); 5434ea2ee4dSMatthew Dillon char *strdup_or_null(char *str); 5448e25f19bSMatthew Dillon void dlogreset(void); 54554f2fefcSMatthew Dillon int dlog00_fd(void); 5466fd67931SMatthew Dillon void addbuildenv(const char *label, const char *data, int type); 547a67bf8dbSMatthew Dillon void delbuildenv(const char *label); 548*aac7a6d9SMatthew Dillon int readlogline(monitorlog_t *log, char **bufp); 5498e25f19bSMatthew Dillon 5508e25f19bSMatthew Dillon void initbulk(void (*func)(bulk_t *bulk), int jobs); 5517f0eca56SMatthew Dillon void queuebulk(const char *s1, const char *s2, const char *s3, 5528e25f19bSMatthew Dillon const char *s4); 5538e25f19bSMatthew Dillon bulk_t *getbulk(void); 5548e25f19bSMatthew Dillon void donebulk(void); 5558e25f19bSMatthew Dillon void freebulk(bulk_t *bulk); 5568e25f19bSMatthew Dillon void freestrp(char **strp); 5578e25f19bSMatthew Dillon void dupstrp(char **strp); 5581645cafeSMatthew Dillon int askyn(const char *ctl, ...); 5593699ee09SMatthew Dillon double getswappct(int *noswapp); 56068dc2eeaSMatthew Dillon FILE *dexec_open(const char **cav, int cac, pid_t *pidp, buildenv_t *xenv, 5616fd67931SMatthew Dillon int with_env, int with_mvars); 5626fd67931SMatthew Dillon int dexec_close(FILE *fp, pid_t pid); 5633cebe4a8SMatthew Dillon const char *getphasestr(worker_phase_t phase); 5648e25f19bSMatthew Dillon 5658e25f19bSMatthew Dillon void ParseConfiguration(int isworker); 5668e25f19bSMatthew Dillon pkg_t *ParsePackageList(int ac, char **av); 5678e25f19bSMatthew Dillon void FreePackageList(pkg_t *pkgs); 5688e25f19bSMatthew Dillon pkg_t *GetLocalPackageList(void); 5698e25f19bSMatthew Dillon pkg_t *GetFullPackageList(void); 5708e25f19bSMatthew Dillon pkg_t *GetPkgPkg(pkg_t *list); 5718e25f19bSMatthew Dillon 5728e25f19bSMatthew Dillon void DoConfigure(void); 5731645cafeSMatthew Dillon void DoStatus(pkg_t *pkgs); 5748e25f19bSMatthew Dillon void DoBuild(pkg_t *pkgs); 5758e25f19bSMatthew Dillon void DoInitBuild(int slot_override); 57687017ac4SMatthew Dillon void DoCleanBuild(int resetlogs); 577a67bf8dbSMatthew Dillon void OptimizeEnv(void); 5788e25f19bSMatthew Dillon void WorkerProcess(int ac, char **av); 5798e25f19bSMatthew Dillon 5801d6e00cdSMatthew Dillon int DoCreateTemplate(int force); 5818e25f19bSMatthew Dillon void DoDestroyTemplate(void); 5828e25f19bSMatthew Dillon void DoWorkerMounts(worker_t *work); 5838e25f19bSMatthew Dillon void DoWorkerUnmounts(worker_t *work); 5848e25f19bSMatthew Dillon void DoRebuildRepo(int ask); 5858e25f19bSMatthew Dillon void DoUpgradePkgs(pkg_t *pkgs, int ask); 5868e25f19bSMatthew Dillon void RemovePackages(pkg_t *pkgs); 5878e25f19bSMatthew Dillon void PurgeDistfiles(pkg_t *pkgs); 5888e25f19bSMatthew Dillon 589ea37671dSMatthew Dillon void RunStatsInit(void); 590ea37671dSMatthew Dillon void RunStatsDone(void); 591ea37671dSMatthew Dillon void RunStatsReset(void); 592*aac7a6d9SMatthew Dillon void RunStatsUpdate(worker_t *work, const char *portdir); 593ea37671dSMatthew Dillon void RunStatsUpdateTop(void); 594ea37671dSMatthew Dillon void RunStatsUpdateLogs(void); 595ea37671dSMatthew Dillon void RunStatsSync(void); 5968e25f19bSMatthew Dillon 5978e25f19bSMatthew Dillon int ipcreadmsg(int fd, wmsg_t *msg); 5988e25f19bSMatthew Dillon int ipcwritemsg(int fd, wmsg_t *msg); 599*aac7a6d9SMatthew Dillon extern void MonitorDirective(const char *datfile, const char *lkfile); 600