17c478bd9Sstevel@tonic-gate /* 27c478bd9Sstevel@tonic-gate * CDDL HEADER START 37c478bd9Sstevel@tonic-gate * 47c478bd9Sstevel@tonic-gate * The contents of this file are subject to the terms of the 5a10acbd6Seschrock * Common Development and Distribution License (the "License"). 6a10acbd6Seschrock * You may not use this file except in compliance with the License. 77c478bd9Sstevel@tonic-gate * 87c478bd9Sstevel@tonic-gate * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 97c478bd9Sstevel@tonic-gate * or http://www.opensolaris.org/os/licensing. 107c478bd9Sstevel@tonic-gate * See the License for the specific language governing permissions 117c478bd9Sstevel@tonic-gate * and limitations under the License. 127c478bd9Sstevel@tonic-gate * 137c478bd9Sstevel@tonic-gate * When distributing Covered Code, include this CDDL HEADER in each 147c478bd9Sstevel@tonic-gate * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 157c478bd9Sstevel@tonic-gate * If applicable, add the following below this CDDL HEADER, with the 167c478bd9Sstevel@tonic-gate * fields enclosed by brackets "[]" replaced with your own identifying 177c478bd9Sstevel@tonic-gate * information: Portions Copyright [yyyy] [name of copyright owner] 187c478bd9Sstevel@tonic-gate * 197c478bd9Sstevel@tonic-gate * CDDL HEADER END 207c478bd9Sstevel@tonic-gate */ 217c478bd9Sstevel@tonic-gate /* 22*f6e214c7SGavin Maltby * Copyright (c) 1998, 2010, Oracle and/or its affiliates. All rights reserved. 237c478bd9Sstevel@tonic-gate */ 247c478bd9Sstevel@tonic-gate 257c478bd9Sstevel@tonic-gate #include <sys/stat.h> 267c478bd9Sstevel@tonic-gate #include <locale.h> 277c478bd9Sstevel@tonic-gate #include <unistd.h> 287c478bd9Sstevel@tonic-gate #include <stdlib.h> 297c478bd9Sstevel@tonic-gate #include <stdio.h> 30*f6e214c7SGavin Maltby #include <string.h> 317c478bd9Sstevel@tonic-gate 327c478bd9Sstevel@tonic-gate #include "dconf.h" 337c478bd9Sstevel@tonic-gate #include "minfree.h" 347c478bd9Sstevel@tonic-gate #include "utils.h" 357c478bd9Sstevel@tonic-gate 367c478bd9Sstevel@tonic-gate static const char USAGE[] = "\ 377c478bd9Sstevel@tonic-gate Usage: %s [-nuy] [-c kernel | curproc | all ] [-d dump-device | swap ]\n\ 38ca3e8d88SDave Plauger [-m min {k|m|%%} ] [-s savecore-dir] [-r root-dir] [-z on|off]\n"; 397c478bd9Sstevel@tonic-gate 40*f6e214c7SGavin Maltby static const char OPTS[] = "inuyc:d:m:s:r:z:"; 417c478bd9Sstevel@tonic-gate 427c478bd9Sstevel@tonic-gate static const char PATH_DEVICE[] = "/dev/dump"; 437c478bd9Sstevel@tonic-gate static const char PATH_CONFIG[] = "/etc/dumpadm.conf"; 447c478bd9Sstevel@tonic-gate 457c478bd9Sstevel@tonic-gate int 467c478bd9Sstevel@tonic-gate main(int argc, char *argv[]) 477c478bd9Sstevel@tonic-gate { 487c478bd9Sstevel@tonic-gate const char *pname = getpname(argv[0]); 497c478bd9Sstevel@tonic-gate 507c478bd9Sstevel@tonic-gate u_longlong_t minf; 517c478bd9Sstevel@tonic-gate struct stat st; 527c478bd9Sstevel@tonic-gate int c; 533e1bd7a2Ssjelinek int dflag = 0; /* for checking in use during -d ops */ 547c478bd9Sstevel@tonic-gate int dcmode = DC_CURRENT; /* kernel settings override unless -u */ 557c478bd9Sstevel@tonic-gate int modified = 0; /* have we modified the dump config? */ 567c478bd9Sstevel@tonic-gate char *minfstr = NULL; /* string value of -m argument */ 577c478bd9Sstevel@tonic-gate dumpconf_t dc; /* current configuration */ 58*f6e214c7SGavin Maltby int chrooted = 0; 59*f6e214c7SGavin Maltby int douuid = 0; 607c478bd9Sstevel@tonic-gate 617c478bd9Sstevel@tonic-gate (void) setlocale(LC_ALL, ""); 627c478bd9Sstevel@tonic-gate (void) textdomain(TEXT_DOMAIN); 637c478bd9Sstevel@tonic-gate 647c478bd9Sstevel@tonic-gate /* 657c478bd9Sstevel@tonic-gate * Take an initial lap through argv hunting for -r root-dir, 667c478bd9Sstevel@tonic-gate * so that we can chroot before opening the configuration file. 677c478bd9Sstevel@tonic-gate * We also handle -u and any bad options at this point. 687c478bd9Sstevel@tonic-gate */ 697c478bd9Sstevel@tonic-gate while (optind < argc) { 707c478bd9Sstevel@tonic-gate while ((c = getopt(argc, argv, OPTS)) != (int)EOF) { 71*f6e214c7SGavin Maltby if (c == 'r' && ++chrooted && chroot(optarg) == -1) 727c478bd9Sstevel@tonic-gate die(gettext("failed to chroot to %s"), optarg); 737c478bd9Sstevel@tonic-gate else if (c == 'u') 747c478bd9Sstevel@tonic-gate dcmode = DC_OVERRIDE; 757c478bd9Sstevel@tonic-gate else if (c == '?') { 767c478bd9Sstevel@tonic-gate (void) fprintf(stderr, gettext(USAGE), pname); 777c478bd9Sstevel@tonic-gate return (E_USAGE); 787c478bd9Sstevel@tonic-gate } 797c478bd9Sstevel@tonic-gate } 807c478bd9Sstevel@tonic-gate 817c478bd9Sstevel@tonic-gate if (optind < argc) { 827c478bd9Sstevel@tonic-gate warn(gettext("illegal argument -- %s\n"), argv[optind]); 837c478bd9Sstevel@tonic-gate (void) fprintf(stderr, gettext(USAGE), pname); 847c478bd9Sstevel@tonic-gate return (E_USAGE); 857c478bd9Sstevel@tonic-gate } 867c478bd9Sstevel@tonic-gate } 877c478bd9Sstevel@tonic-gate 887c478bd9Sstevel@tonic-gate if (geteuid() != 0) 897c478bd9Sstevel@tonic-gate die(gettext("you must be root to use %s\n"), pname); 907c478bd9Sstevel@tonic-gate 917c478bd9Sstevel@tonic-gate /* 927c478bd9Sstevel@tonic-gate * If no config file exists yet, we're going to create an empty one, 937c478bd9Sstevel@tonic-gate * so set the modified flag to force writing out the file. 947c478bd9Sstevel@tonic-gate */ 957c478bd9Sstevel@tonic-gate if (access(PATH_CONFIG, F_OK) == -1) 967c478bd9Sstevel@tonic-gate modified++; 977c478bd9Sstevel@tonic-gate 987c478bd9Sstevel@tonic-gate /* 997c478bd9Sstevel@tonic-gate * Now open and read in the initial values from the config file. 1007c478bd9Sstevel@tonic-gate * If it doesn't exist, we create an empty file and dc is 1017c478bd9Sstevel@tonic-gate * initialized with the default values. 1027c478bd9Sstevel@tonic-gate */ 1037c478bd9Sstevel@tonic-gate if (dconf_open(&dc, PATH_DEVICE, PATH_CONFIG, dcmode) == -1) 1047c478bd9Sstevel@tonic-gate return (E_ERROR); 1057c478bd9Sstevel@tonic-gate 1067c478bd9Sstevel@tonic-gate /* 1077c478bd9Sstevel@tonic-gate * Take another lap through argv, processing options and 1087c478bd9Sstevel@tonic-gate * modifying the dumpconf_t as appropriate. 1097c478bd9Sstevel@tonic-gate */ 1107c478bd9Sstevel@tonic-gate for (optind = 1; optind < argc; optind++) { 1117c478bd9Sstevel@tonic-gate while ((c = getopt(argc, argv, OPTS)) != (int)EOF) { 1127c478bd9Sstevel@tonic-gate switch (c) { 1137c478bd9Sstevel@tonic-gate case 'c': 1147c478bd9Sstevel@tonic-gate if (dconf_str2content(&dc, optarg) == -1) 1157c478bd9Sstevel@tonic-gate return (E_USAGE); 1167c478bd9Sstevel@tonic-gate modified++; 1177c478bd9Sstevel@tonic-gate break; 1187c478bd9Sstevel@tonic-gate case 'd': 1197c478bd9Sstevel@tonic-gate if (dconf_str2device(&dc, optarg) == -1) 1207c478bd9Sstevel@tonic-gate return (E_USAGE); 1213e1bd7a2Ssjelinek dflag++; 1227c478bd9Sstevel@tonic-gate modified++; 1237c478bd9Sstevel@tonic-gate break; 1247c478bd9Sstevel@tonic-gate 125*f6e214c7SGavin Maltby case 'i': 126*f6e214c7SGavin Maltby /* undocumented option */ 127*f6e214c7SGavin Maltby if (chrooted) { 128*f6e214c7SGavin Maltby warn(gettext("-i and -r cannot be " 129*f6e214c7SGavin Maltby "used together\n")); 130*f6e214c7SGavin Maltby return (E_USAGE); 131*f6e214c7SGavin Maltby } 132*f6e214c7SGavin Maltby douuid++; 133*f6e214c7SGavin Maltby break; 134*f6e214c7SGavin Maltby 1357c478bd9Sstevel@tonic-gate case 'm': 1367c478bd9Sstevel@tonic-gate minfstr = optarg; 1377c478bd9Sstevel@tonic-gate break; 1387c478bd9Sstevel@tonic-gate 1397c478bd9Sstevel@tonic-gate case 'n': 1407c478bd9Sstevel@tonic-gate dc.dc_enable = DC_OFF; 1417c478bd9Sstevel@tonic-gate modified++; 1427c478bd9Sstevel@tonic-gate break; 1437c478bd9Sstevel@tonic-gate 1447c478bd9Sstevel@tonic-gate case 's': 1457c478bd9Sstevel@tonic-gate if (stat(optarg, &st) == -1 || 1467c478bd9Sstevel@tonic-gate !S_ISDIR(st.st_mode)) { 1477c478bd9Sstevel@tonic-gate warn(gettext("%s is missing or not a " 1487c478bd9Sstevel@tonic-gate "directory\n"), optarg); 1497c478bd9Sstevel@tonic-gate return (E_USAGE); 1507c478bd9Sstevel@tonic-gate } 1517c478bd9Sstevel@tonic-gate 1527c478bd9Sstevel@tonic-gate if (dconf_str2savdir(&dc, optarg) == -1) 1537c478bd9Sstevel@tonic-gate return (E_USAGE); 1547c478bd9Sstevel@tonic-gate modified++; 1557c478bd9Sstevel@tonic-gate break; 1567c478bd9Sstevel@tonic-gate 1577c478bd9Sstevel@tonic-gate case 'y': 1587c478bd9Sstevel@tonic-gate dc.dc_enable = DC_ON; 1597c478bd9Sstevel@tonic-gate modified++; 1607c478bd9Sstevel@tonic-gate break; 161ca3e8d88SDave Plauger 162ca3e8d88SDave Plauger case 'z': 163ca3e8d88SDave Plauger if (dconf_str2csave(&dc, optarg) == -1) 164ca3e8d88SDave Plauger return (E_USAGE); 165ca3e8d88SDave Plauger modified++; 166ca3e8d88SDave Plauger break; 1677c478bd9Sstevel@tonic-gate } 1687c478bd9Sstevel@tonic-gate } 1697c478bd9Sstevel@tonic-gate } 1707c478bd9Sstevel@tonic-gate 171*f6e214c7SGavin Maltby if (douuid) 172*f6e214c7SGavin Maltby return (dconf_write_uuid(&dc) ? E_SUCCESS : E_ERROR); 173*f6e214c7SGavin Maltby 1747c478bd9Sstevel@tonic-gate if (minfstr != NULL) { 1757c478bd9Sstevel@tonic-gate if (minfree_compute(dc.dc_savdir, minfstr, &minf) == -1) 1767c478bd9Sstevel@tonic-gate return (E_USAGE); 1777c478bd9Sstevel@tonic-gate if (minfree_write(dc.dc_savdir, minf) == -1) 1787c478bd9Sstevel@tonic-gate return (E_ERROR); 1797c478bd9Sstevel@tonic-gate } 1807c478bd9Sstevel@tonic-gate 1817c478bd9Sstevel@tonic-gate if (dcmode == DC_OVERRIDE) { 1827c478bd9Sstevel@tonic-gate /* 1837c478bd9Sstevel@tonic-gate * In override mode, we try to force an update. If this 1847c478bd9Sstevel@tonic-gate * fails, we re-load the kernel configuration and write that 1857c478bd9Sstevel@tonic-gate * out to the file in order to force the file in sync. 186a10acbd6Seschrock * 187a10acbd6Seschrock * We allow the file to be read-only but print a warning to the 188a10acbd6Seschrock * user that indicates it hasn't been updated. 1897c478bd9Sstevel@tonic-gate */ 1903e1bd7a2Ssjelinek if (dconf_update(&dc, 0) == -1) 1917c478bd9Sstevel@tonic-gate (void) dconf_getdev(&dc); 192a10acbd6Seschrock if (dc.dc_readonly) 193a10acbd6Seschrock warn(gettext("kernel settings updated, but " 194a10acbd6Seschrock "%s is read-only\n"), PATH_CONFIG); 195a10acbd6Seschrock else if (dconf_write(&dc) == -1) 1967c478bd9Sstevel@tonic-gate return (E_ERROR); 1977c478bd9Sstevel@tonic-gate 1987c478bd9Sstevel@tonic-gate } else if (modified) { 1997c478bd9Sstevel@tonic-gate /* 2007c478bd9Sstevel@tonic-gate * If we're modifying the configuration, then try 2017c478bd9Sstevel@tonic-gate * to update it, and write out the file if successful. 2027c478bd9Sstevel@tonic-gate */ 203a10acbd6Seschrock if (dc.dc_readonly) { 204a10acbd6Seschrock warn(gettext("failed to update settings: %s is " 205a10acbd6Seschrock "read-only\n"), PATH_CONFIG); 206a10acbd6Seschrock return (E_ERROR); 207a10acbd6Seschrock } 208a10acbd6Seschrock 2093e1bd7a2Ssjelinek if (dconf_update(&dc, dflag) == -1 || 2103e1bd7a2Ssjelinek dconf_write(&dc) == -1) 2117c478bd9Sstevel@tonic-gate return (E_ERROR); 2127c478bd9Sstevel@tonic-gate } 2137c478bd9Sstevel@tonic-gate 2147c478bd9Sstevel@tonic-gate if (dcmode == DC_CURRENT) 2157c478bd9Sstevel@tonic-gate dconf_print(&dc, stdout); 2167c478bd9Sstevel@tonic-gate 2177c478bd9Sstevel@tonic-gate if (dconf_close(&dc) == -1) 2187c478bd9Sstevel@tonic-gate warn(gettext("failed to close configuration file")); 2197c478bd9Sstevel@tonic-gate 2207c478bd9Sstevel@tonic-gate return (E_SUCCESS); 2217c478bd9Sstevel@tonic-gate } 222