/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License (the "License"). * You may not use this file except in compliance with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright (c) 2008, 2010, Oracle and/or its affiliates. All rights reserved. * Copyright 2013 Nexenta Systems, Inc. All rights reserved. * Copyright 2016 Toomas Soome * Copyright (c) 2015 by Delphix. All rights reserved. * Copyright 2018 OmniOS Community Edition (OmniOSce) Association. */ /* * System includes */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include /* Private function prototypes */ static int update_dataset(char *, int, char *, char *, char *); static int _update_vfstab(char *, char *, char *, char *, be_fs_list_data_t *); static int be_open_menu(char *, char *, FILE **, char *, boolean_t); static int be_create_menu(char *, char *, FILE **, char *); static char *be_get_auto_name(char *, char *, boolean_t); /* * Global error printing */ boolean_t do_print = B_FALSE; /* * Private datatypes */ typedef struct zone_be_name_cb_data { char *base_be_name; int num; } zone_be_name_cb_data_t; /* ******************************************************************** */ /* Public Functions */ /* ******************************************************************** */ /* * Callback for ficl to suppress all output from ficl, as we do not * want to confuse user with messages from ficl, and we are only * checking results from function calls. */ /*ARGSUSED*/ static void ficlSuppressTextOutput(ficlCallback *cb, char *text) { /* This function is intentionally doing nothing. */ } /* * Function: be_get_boot_args * Description: Returns the fast boot argument string for enumerated BE. * Parameters: * fbarg - pointer to argument string. * entry - index of BE. * Returns: * fast boot argument string. * Scope: * Public */ int be_get_boot_args(char **fbarg, int entry) { be_node_list_t *node, *be_nodes = NULL; be_transaction_data_t bt = {0}; char *mountpoint = NULL; boolean_t be_mounted = B_FALSE; int ret = BE_SUCCESS; int index; ficlVm *vm; *fbarg = NULL; if (!be_zfs_init()) return (BE_ERR_INIT); /* * need pool name, menu.lst has entries from our pool only */ ret = be_find_current_be(&bt); if (ret != BE_SUCCESS) { be_zfs_fini(); return (ret); } /* * be_get_boot_args() is for loader, fail with grub will trigger * normal boot. */ if (be_has_grub()) { ret = BE_ERR_INIT; goto done; } ret = _be_list(NULL, &be_nodes, BE_LIST_DEFAULT); if (ret != BE_SUCCESS) goto done; /* * iterate through be_nodes, * if entry == -1, stop if be_active_on_boot, * else stop if index == entry. */ index = 0; for (node = be_nodes; node != NULL; node = node->be_next_node) { if (strcmp(node->be_rpool, bt.obe_zpool) != 0) continue; if (entry == BE_ENTRY_DEFAULT && node->be_active_on_boot == B_TRUE) break; if (index == entry) break; index++; } if (node == NULL) { be_free_list(be_nodes); ret = BE_ERR_NOENT; goto done; } /* try to mount inactive be */ if (node->be_active == B_FALSE) { ret = _be_mount(node->be_node_name, &mountpoint, BE_MOUNT_FLAG_NO_ZONES); if (ret != BE_SUCCESS && ret != BE_ERR_MOUNTED) { be_free_list(be_nodes); goto done; } else be_mounted = B_TRUE; } vm = bf_init("", ficlSuppressTextOutput); if (vm != NULL) { /* * zfs MAXNAMELEN is 256, so we need to pick buf large enough * to contain such names. */ char buf[MAXNAMELEN * 2]; char *kernel_options = NULL; char *kernel = NULL; char *tmp; zpool_handle_t *zph; /* * just try to interpret following words. on error * we will be missing kernelname, and will get out. */ (void) snprintf(buf, sizeof (buf), "set currdev=zfs:%s:", node->be_root_ds); ret = ficlVmEvaluate(vm, buf); if (ret != FICL_VM_STATUS_OUT_OF_TEXT) { be_print_err(gettext("be_get_boot_args: error " "interpreting boot config: %d\n"), ret); bf_fini(); ret = BE_ERR_NO_MENU; goto cleanup; } (void) snprintf(buf, sizeof (buf), "include /boot/forth/loader.4th"); ret = ficlVmEvaluate(vm, buf); if (ret != FICL_VM_STATUS_OUT_OF_TEXT) { be_print_err(gettext("be_get_boot_args: error " "interpreting boot config: %d\n"), ret); bf_fini(); ret = BE_ERR_NO_MENU; goto cleanup; } (void) snprintf(buf, sizeof (buf), "start"); ret = ficlVmEvaluate(vm, buf); if (ret != FICL_VM_STATUS_OUT_OF_TEXT) { be_print_err(gettext("be_get_boot_args: error " "interpreting boot config: %d\n"), ret); bf_fini(); ret = BE_ERR_NO_MENU; goto cleanup; } (void) snprintf(buf, sizeof (buf), "boot"); ret = ficlVmEvaluate(vm, buf); bf_fini(); if (ret != FICL_VM_STATUS_OUT_OF_TEXT) { be_print_err(gettext("be_get_boot_args: error " "interpreting boot config: %d\n"), ret); ret = BE_ERR_NO_MENU; goto cleanup; } kernel_options = getenv("boot-args"); kernel = getenv("kernelname"); if (kernel == NULL) { be_print_err(gettext("be_get_boot_args: no kernel\n")); ret = BE_ERR_NOENT; goto cleanup; } if ((zph = zpool_open(g_zfs, node->be_rpool)) == NULL) { be_print_err(gettext("be_get_boot_args: failed to " "open root pool (%s): %s\n"), node->be_rpool, libzfs_error_description(g_zfs)); ret = zfs_err_to_be_err(g_zfs); goto cleanup; } ret = zpool_get_physpath(zph, buf, sizeof (buf)); zpool_close(zph); if (ret != 0) { be_print_err(gettext("be_get_boot_args: failed to " "get physpath\n")); goto cleanup; } /* zpool_get_physpath() can return space separated list */ tmp = buf; tmp = strsep(&tmp, " "); if (kernel_options == NULL || *kernel_options == '\0') (void) asprintf(fbarg, "/ %s " "-B zfs-bootfs=%s,bootpath=\"%s\"\n", kernel, node->be_root_ds, tmp); else (void) asprintf(fbarg, "/ %s %s " "-B zfs-bootfs=%s,bootpath=\"%s\"\n", kernel, kernel_options, node->be_root_ds, tmp); if (fbarg == NULL) ret = BE_ERR_NOMEM; else ret = 0; } else ret = BE_ERR_NOMEM; cleanup: if (be_mounted == B_TRUE) (void) _be_unmount(node->be_node_name, BE_UNMOUNT_FLAG_FORCE); be_free_list(be_nodes); done: free(mountpoint); free(bt.obe_name); free(bt.obe_root_ds); free(bt.obe_zpool); free(bt.obe_snap_name); free(bt.obe_altroot); be_zfs_fini(); return (ret); } /* * Function: be_max_avail * Description: Returns the available size for the zfs dataset passed in. * Parameters: * dataset - The dataset we want to get the available space for. * ret - The available size will be returned in this. * Returns: * The error returned by the zfs get property function. * Scope: * Public */ int be_max_avail(char *dataset, uint64_t *ret) { zfs_handle_t *zhp; int err = 0; /* Initialize libzfs handle */ if (!be_zfs_init()) return (BE_ERR_INIT); zhp = zfs_open(g_zfs, dataset, ZFS_TYPE_DATASET); if (zhp == NULL) { /* * The zfs_open failed return an error */ err = zfs_err_to_be_err(g_zfs); } else { err = be_maxsize_avail(zhp, ret); } ZFS_CLOSE(zhp); be_zfs_fini(); return (err); } /* * Function: libbe_print_errors * Description: Turns on/off error output for the library. * Parameter: * set_do_print - Boolean that turns library error * printing on or off. * Returns: * None * Scope: * Public; */ void libbe_print_errors(boolean_t set_do_print) { do_print = set_do_print; } /* ******************************************************************** */ /* Semi-Private Functions */ /* ******************************************************************** */ /* * Function: be_zfs_init * Description: Initializes the libary global libzfs handle. * Parameters: * None * Returns: * B_TRUE - Success * B_FALSE - Failure * Scope: * Semi-private (library wide use only) */ boolean_t be_zfs_init(void) { be_zfs_fini(); if ((g_zfs = libzfs_init()) == NULL) { be_print_err(gettext("be_zfs_init: failed to initialize ZFS " "library\n")); return (B_FALSE); } return (B_TRUE); } /* * Function: be_zfs_fini * Description: Closes the library global libzfs handle if it currently open. * Parameter: * None * Returns: * None * Scope: * Semi-private (library wide use only) */ void be_zfs_fini(void) { if (g_zfs) libzfs_fini(g_zfs); g_zfs = NULL; } /* * Function: be_get_defaults * Description: Open defaults and gets be default paramets * Parameters: * defaults - be defaults struct * Returns: * None * Scope: * Semi-private (library wide use only) */ void be_get_defaults(struct be_defaults *defaults) { void *defp; defaults->be_deflt_grub = B_FALSE; defaults->be_deflt_rpool_container = B_FALSE; defaults->be_deflt_bename_starts_with[0] = '\0'; if ((defp = defopen_r(BE_DEFAULTS)) != NULL) { const char *res = defread_r(BE_DFLT_BENAME_STARTS, defp); if (res != NULL && res[0] != '\0') { (void) strlcpy(defaults->be_deflt_bename_starts_with, res, ZFS_MAX_DATASET_NAME_LEN); defaults->be_deflt_rpool_container = B_TRUE; } if (be_is_isa("i386")) { res = defread_r(BE_DFLT_BE_HAS_GRUB, defp); if (res != NULL && res[0] != '\0') { if (strcasecmp(res, "true") == 0) defaults->be_deflt_grub = B_TRUE; } } defclose_r(defp); } } /* * Function: be_make_root_ds * Description: Generate string for BE's root dataset given the pool * it lives in and the BE name. * Parameters: * zpool - pointer zpool name. * be_name - pointer to BE name. * be_root_ds - pointer to buffer to return BE root dataset in. * be_root_ds_size - size of be_root_ds * Returns: * None * Scope: * Semi-private (library wide use only) */ void be_make_root_ds(const char *zpool, const char *be_name, char *be_root_ds, int be_root_ds_size) { struct be_defaults be_defaults; be_get_defaults(&be_defaults); char *root_ds = NULL; if (getzoneid() == GLOBAL_ZONEID) { if (be_defaults.be_deflt_rpool_container) { (void) snprintf(be_root_ds, be_root_ds_size, "%s/%s", zpool, be_name); } else { (void) snprintf(be_root_ds, be_root_ds_size, "%s/%s/%s", zpool, BE_CONTAINER_DS_NAME, be_name); } } else { /* * In non-global zone we can use path from mounted root dataset * to generate BE's root dataset string. */ if ((root_ds = be_get_ds_from_dir("/")) != NULL) { (void) snprintf(be_root_ds, be_root_ds_size, "%s/%s", dirname(root_ds), be_name); } else { be_print_err(gettext("be_make_root_ds: zone root " "dataset is not mounted\n")); return; } } } /* * Function: be_make_container_ds * Description: Generate string for the BE container dataset given a pool name. * Parameters: * zpool - pointer zpool name. * container_ds - pointer to buffer to return BE container * dataset in. * container_ds_size - size of container_ds * Returns: * None * Scope: * Semi-private (library wide use only) */ void be_make_container_ds(const char *zpool, char *container_ds, int container_ds_size) { struct be_defaults be_defaults; be_get_defaults(&be_defaults); char *root_ds = NULL; if (getzoneid() == GLOBAL_ZONEID) { if (be_defaults.be_deflt_rpool_container) { (void) snprintf(container_ds, container_ds_size, "%s", zpool); } else { (void) snprintf(container_ds, container_ds_size, "%s/%s", zpool, BE_CONTAINER_DS_NAME); } } else { if ((root_ds = be_get_ds_from_dir("/")) != NULL) { (void) strlcpy(container_ds, dirname(root_ds), container_ds_size); } else { be_print_err(gettext("be_make_container_ds: zone root " "dataset is not mounted\n")); return; } } } /* * Function: be_make_name_from_ds * Description: This function takes a dataset name and strips off the * BE container dataset portion from the beginning. The * returned name is allocated in heap storage, so the caller * is responsible for freeing it. * Parameters: * dataset - dataset to get name from. * rc_loc - dataset underwhich the root container dataset lives. * Returns: * name of dataset relative to BE container dataset. * NULL if dataset is not under a BE root dataset. * Scope: * Semi-primate (library wide use only) */ char * be_make_name_from_ds(const char *dataset, char *rc_loc) { char ds[ZFS_MAX_DATASET_NAME_LEN]; char *tok = NULL; char *name = NULL; struct be_defaults be_defaults; int rlen = strlen(rc_loc); be_get_defaults(&be_defaults); /* * First token is the location of where the root container dataset * lives; it must match rc_loc. */ if (strncmp(dataset, rc_loc, rlen) == 0 && dataset[rlen] == '/') (void) strlcpy(ds, dataset + rlen + 1, sizeof (ds)); else return (NULL); if (be_defaults.be_deflt_rpool_container) { if ((name = strdup(ds)) == NULL) { be_print_err(gettext("be_make_name_from_ds: " "memory allocation failed\n")); return (NULL); } } else { /* Second token must be BE container dataset name */ if ((tok = strtok(ds, "/")) == NULL || strcmp(tok, BE_CONTAINER_DS_NAME) != 0) return (NULL); /* Return the remaining token if one exists */ if ((tok = strtok(NULL, "")) == NULL) return (NULL); if ((name = strdup(tok)) == NULL) { be_print_err(gettext("be_make_name_from_ds: " "memory allocation failed\n")); return (NULL); } } return (name); } /* * Function: be_maxsize_avail * Description: Returns the available size for the zfs handle passed in. * Parameters: * zhp - A pointer to the open zfs handle. * ret - The available size will be returned in this. * Returns: * The error returned by the zfs get property function. * Scope: * Semi-private (library wide use only) */ int be_maxsize_avail(zfs_handle_t *zhp, uint64_t *ret) { return ((*ret = zfs_prop_get_int(zhp, ZFS_PROP_AVAILABLE))); } /* * Function: be_append_menu * Description: Appends an entry for a BE into the menu.lst. * Parameters: * be_name - pointer to name of BE to add boot menu entry for. * be_root_pool - pointer to name of pool BE lives in. * boot_pool - Used if the pool containing the grub menu is * different than the one contaiing the BE. This * will normally be NULL. * be_orig_root_ds - The root dataset for the BE. This is * used to check to see if an entry already exists * for this BE. * description - pointer to description of BE to be added in * the title line for this BEs entry. * Returns: * BE_SUCCESS - Success * be_errno_t - Failure * Scope: * Semi-private (library wide use only) */ int be_append_menu(char *be_name, char *be_root_pool, char *boot_pool, char *be_orig_root_ds, char *description) { zfs_handle_t *zhp = NULL; char menu_file[MAXPATHLEN]; char be_root_ds[MAXPATHLEN]; char line[BUFSIZ]; char temp_line[BUFSIZ]; char title[MAXPATHLEN]; char *entries[BUFSIZ]; char *tmp_entries[BUFSIZ]; char *pool_mntpnt = NULL; char *ptmp_mntpnt = NULL; char *orig_mntpnt = NULL; boolean_t found_be = B_FALSE; boolean_t found_orig_be = B_FALSE; boolean_t found_title = B_FALSE; boolean_t pool_mounted = B_FALSE; boolean_t collect_lines = B_FALSE; FILE *menu_fp = NULL; int err = 0, ret = BE_SUCCESS; int i, num_tmp_lines = 0, num_lines = 0; if (be_name == NULL || be_root_pool == NULL) return (BE_ERR_INVAL); if (boot_pool == NULL) boot_pool = be_root_pool; if ((zhp = zfs_open(g_zfs, be_root_pool, ZFS_TYPE_DATASET)) == NULL) { be_print_err(gettext("be_append_menu: failed to open " "pool dataset for %s: %s\n"), be_root_pool, libzfs_error_description(g_zfs)); return (zfs_err_to_be_err(g_zfs)); } /* * Check to see if the pool's dataset is mounted. If it isn't we'll * attempt to mount it. */ if ((ret = be_mount_pool(zhp, &ptmp_mntpnt, &orig_mntpnt, &pool_mounted)) != BE_SUCCESS) { be_print_err(gettext("be_append_menu: pool dataset " "(%s) could not be mounted\n"), be_root_pool); ZFS_CLOSE(zhp); return (ret); } /* * Get the mountpoint for the root pool dataset. */ if (!zfs_is_mounted(zhp, &pool_mntpnt)) { be_print_err(gettext("be_append_menu: pool " "dataset (%s) is not mounted. Can't set " "the default BE in the grub menu.\n"), be_root_pool); ret = BE_ERR_NO_MENU; goto cleanup; } /* * Check to see if this system supports grub */ if (be_has_grub()) { (void) snprintf(menu_file, sizeof (menu_file), "%s%s", pool_mntpnt, BE_GRUB_MENU); } else { (void) snprintf(menu_file, sizeof (menu_file), "%s%s", pool_mntpnt, BE_SPARC_MENU); } be_make_root_ds(be_root_pool, be_name, be_root_ds, sizeof (be_root_ds)); /* * Iterate through menu first to make sure the BE doesn't already * have an entry in the menu. * * Additionally while iterating through the menu, if we have an * original root dataset for a BE we're cloning from, we need to keep * track of that BE's menu entry. We will then use the lines from * that entry to create the entry for the new BE. */ if ((ret = be_open_menu(be_root_pool, menu_file, &menu_fp, "r", B_TRUE)) != BE_SUCCESS) { goto cleanup; } else if (menu_fp == NULL) { ret = BE_ERR_NO_MENU; goto cleanup; } free(pool_mntpnt); pool_mntpnt = NULL; while (fgets(line, BUFSIZ, menu_fp)) { char *tok = NULL; (void) strlcpy(temp_line, line, BUFSIZ); tok = strtok(line, BE_WHITE_SPACE); if (tok == NULL || tok[0] == '#') { continue; } else if (strcmp(tok, "title") == 0) { collect_lines = B_FALSE; if ((tok = strtok(NULL, "\n")) == NULL) (void) strlcpy(title, "", sizeof (title)); else (void) strlcpy(title, tok, sizeof (title)); found_title = B_TRUE; if (num_tmp_lines != 0) { for (i = 0; i < num_tmp_lines; i++) { free(tmp_entries[i]); tmp_entries[i] = NULL; } num_tmp_lines = 0; } } else if (strcmp(tok, "bootfs") == 0) { char *bootfs = strtok(NULL, BE_WHITE_SPACE); found_title = B_FALSE; if (bootfs == NULL) continue; if (strcmp(bootfs, be_root_ds) == 0) { found_be = B_TRUE; break; } if (be_orig_root_ds != NULL && strcmp(bootfs, be_orig_root_ds) == 0 && !found_orig_be) { char str[BUFSIZ]; found_orig_be = B_TRUE; num_lines = 0; /* * Store the new title line */ (void) snprintf(str, BUFSIZ, "title %s\n", description ? description : be_name); entries[num_lines] = strdup(str); num_lines++; /* * If there are any lines between the title * and the bootfs line store these. Also * free the temporary lines. */ for (i = 0; i < num_tmp_lines; i++) { entries[num_lines] = tmp_entries[i]; tmp_entries[i] = NULL; num_lines++; } num_tmp_lines = 0; /* * Store the new bootfs line. */ (void) snprintf(str, BUFSIZ, "bootfs %s\n", be_root_ds); entries[num_lines] = strdup(str); num_lines++; collect_lines = B_TRUE; } } else if (found_orig_be && collect_lines) { /* * get the rest of the lines for the original BE and * store them. */ if (strstr(line, BE_GRUB_COMMENT) != NULL || strstr(line, "BOOTADM") != NULL) continue; if (strcmp(tok, "splashimage") == 0) { entries[num_lines] = strdup("splashimage " "/boot/splashimage.xpm\n"); } else { entries[num_lines] = strdup(temp_line); } num_lines++; } else if (found_title && !found_orig_be) { tmp_entries[num_tmp_lines] = strdup(temp_line); num_tmp_lines++; } } (void) fclose(menu_fp); if (found_be) { /* * If an entry for this BE was already in the menu, then if * that entry's title matches what we would have put in * return success. Otherwise return failure. */ char *new_title = description ? description : be_name; if (strcmp(title, new_title) == 0) { ret = BE_SUCCESS; goto cleanup; } else { if (be_remove_menu(be_name, be_root_pool, boot_pool) != BE_SUCCESS) { be_print_err(gettext("be_append_menu: " "Failed to remove existing unusable " "entry '%s' in boot menu.\n"), be_name); ret = BE_ERR_BE_EXISTS; goto cleanup; } } } /* Append BE entry to the end of the file */ menu_fp = fopen(menu_file, "a+"); err = errno; if (menu_fp == NULL) { be_print_err(gettext("be_append_menu: failed " "to open menu.lst file %s\n"), menu_file); ret = errno_to_be_err(err); goto cleanup; } if (found_orig_be) { /* * write out all the stored lines */ for (i = 0; i < num_lines; i++) { (void) fprintf(menu_fp, "%s", entries[i]); free(entries[i]); } num_lines = 0; /* * Check to see if this system supports grub */ if (be_has_grub()) (void) fprintf(menu_fp, "%s\n", BE_GRUB_COMMENT); ret = BE_SUCCESS; } else { (void) fprintf(menu_fp, "title %s\n", description ? description : be_name); (void) fprintf(menu_fp, "bootfs %s\n", be_root_ds); /* * Check to see if this system supports grub */ if (be_has_grub()) { (void) fprintf(menu_fp, "kernel$ " "/platform/i86pc/kernel/$ISADIR/unix -B " "$ZFS-BOOTFS\n"); (void) fprintf(menu_fp, "module$ " "/platform/i86pc/$ISADIR/boot_archive\n"); (void) fprintf(menu_fp, "%s\n", BE_GRUB_COMMENT); } ret = BE_SUCCESS; } (void) fclose(menu_fp); cleanup: if (pool_mounted) { int err = BE_SUCCESS; err = be_unmount_pool(zhp, ptmp_mntpnt, orig_mntpnt); if (ret == BE_SUCCESS) ret = err; free(orig_mntpnt); free(ptmp_mntpnt); } ZFS_CLOSE(zhp); if (num_tmp_lines > 0) { for (i = 0; i < num_tmp_lines; i++) { free(tmp_entries[i]); tmp_entries[i] = NULL; } } if (num_lines > 0) { for (i = 0; i < num_lines; i++) { free(entries[i]); entries[i] = NULL; } } return (ret); } /* * Function: be_remove_menu * Description: Removes a BE's entry from a menu.lst file. * Parameters: * be_name - the name of BE whose entry is to be removed from * the menu.lst file. * be_root_pool - the pool that be_name lives in. * boot_pool - the pool where the BE is, if different than * the pool containing the boot menu. If this is * NULL it will be set to be_root_pool. * Returns: * BE_SUCCESS - Success * be_errno_t - Failure * Scope: * Semi-private (library wide use only) */ int be_remove_menu(char *be_name, char *be_root_pool, char *boot_pool) { zfs_handle_t *zhp = NULL; char be_root_ds[MAXPATHLEN]; char **buffer = NULL; char menu_buf[BUFSIZ]; char menu[MAXPATHLEN]; char *pool_mntpnt = NULL; char *ptmp_mntpnt = NULL; char *orig_mntpnt = NULL; char *tmp_menu = NULL; FILE *menu_fp = NULL; FILE *tmp_menu_fp = NULL; struct stat sb; int ret = BE_SUCCESS; int i; int fd; int err = 0; int nlines = 0; int default_entry = 0; int entry_cnt = 0; int entry_del = 0; int num_entry_del = 0; int tmp_menu_len = 0; boolean_t write = B_TRUE; boolean_t do_buffer = B_FALSE; boolean_t pool_mounted = B_FALSE; if (boot_pool == NULL) boot_pool = be_root_pool; /* Get name of BE's root dataset */ be_make_root_ds(be_root_pool, be_name, be_root_ds, sizeof (be_root_ds)); /* Get handle to pool dataset */ if ((zhp = zfs_open(g_zfs, be_root_pool, ZFS_TYPE_DATASET)) == NULL) { be_print_err(gettext("be_remove_menu: " "failed to open pool dataset for %s: %s"), be_root_pool, libzfs_error_description(g_zfs)); return (zfs_err_to_be_err(g_zfs)); } /* * Check to see if the pool's dataset is mounted. If it isn't we'll * attempt to mount it. */ if ((ret = be_mount_pool(zhp, &ptmp_mntpnt, &orig_mntpnt, &pool_mounted)) != BE_SUCCESS) { be_print_err(gettext("be_remove_menu: pool dataset " "(%s) could not be mounted\n"), be_root_pool); ZFS_CLOSE(zhp); return (ret); } /* * Get the mountpoint for the root pool dataset. */ if (!zfs_is_mounted(zhp, &pool_mntpnt)) { be_print_err(gettext("be_remove_menu: pool " "dataset (%s) is not mounted. Can't set " "the default BE in the grub menu.\n"), be_root_pool); ret = BE_ERR_NO_MENU; goto cleanup; } /* Get path to boot menu */ (void) strlcpy(menu, pool_mntpnt, sizeof (menu)); /* * Check to see if this system supports grub */ if (be_has_grub()) (void) strlcat(menu, BE_GRUB_MENU, sizeof (menu)); else (void) strlcat(menu, BE_SPARC_MENU, sizeof (menu)); /* Get handle to boot menu file */ if ((ret = be_open_menu(be_root_pool, menu, &menu_fp, "r", B_TRUE)) != BE_SUCCESS) { goto cleanup; } else if (menu_fp == NULL) { ret = BE_ERR_NO_MENU; goto cleanup; } free(pool_mntpnt); pool_mntpnt = NULL; /* Grab the stats of the original menu file */ if (stat(menu, &sb) != 0) { err = errno; be_print_err(gettext("be_remove_menu: " "failed to stat file %s: %s\n"), menu, strerror(err)); ret = errno_to_be_err(err); goto cleanup; } /* Create a tmp file for the modified menu.lst */ tmp_menu_len = strlen(menu) + 7; if ((tmp_menu = (char *)malloc(tmp_menu_len)) == NULL) { be_print_err(gettext("be_remove_menu: malloc failed\n")); ret = BE_ERR_NOMEM; goto cleanup; } (void) memset(tmp_menu, 0, tmp_menu_len); (void) strlcpy(tmp_menu, menu, tmp_menu_len); (void) strlcat(tmp_menu, "XXXXXX", tmp_menu_len); if ((fd = mkstemp(tmp_menu)) == -1) { err = errno; be_print_err(gettext("be_remove_menu: mkstemp failed\n")); ret = errno_to_be_err(err); free(tmp_menu); tmp_menu = NULL; goto cleanup; } if ((tmp_menu_fp = fdopen(fd, "w")) == NULL) { err = errno; be_print_err(gettext("be_remove_menu: " "could not open tmp file for write: %s\n"), strerror(err)); (void) close(fd); ret = errno_to_be_err(err); goto cleanup; } while (fgets(menu_buf, BUFSIZ, menu_fp)) { char tline [BUFSIZ]; char *tok = NULL; (void) strlcpy(tline, menu_buf, sizeof (tline)); /* Tokenize line */ tok = strtok(tline, BE_WHITE_SPACE); if (tok == NULL || tok[0] == '#') { /* Found empty line or comment line */ if (do_buffer) { /* Buffer this line */ if ((buffer = (char **)realloc(buffer, sizeof (char *)*(nlines + 1))) == NULL) { ret = BE_ERR_NOMEM; goto cleanup; } if ((buffer[nlines++] = strdup(menu_buf)) == NULL) { ret = BE_ERR_NOMEM; goto cleanup; } } else if (write || strncmp(menu_buf, BE_GRUB_COMMENT, strlen(BE_GRUB_COMMENT)) != 0) { /* Write this line out */ (void) fputs(menu_buf, tmp_menu_fp); } } else if (strcmp(tok, "default") == 0) { /* * Record what 'default' is set to because we might * need to adjust this upon deleting an entry. */ tok = strtok(NULL, BE_WHITE_SPACE); if (tok != NULL) { default_entry = atoi(tok); } (void) fputs(menu_buf, tmp_menu_fp); } else if (strcmp(tok, "title") == 0) { /* * If we've reached a 'title' line and do_buffer is * is true, that means we've just buffered an entire * entry without finding a 'bootfs' directive. We * need to write that entry out and keep searching. */ if (do_buffer) { for (i = 0; i < nlines; i++) { (void) fputs(buffer[i], tmp_menu_fp); free(buffer[i]); } free(buffer); buffer = NULL; nlines = 0; } /* * Turn writing off and buffering on, and increment * our entry counter. */ write = B_FALSE; do_buffer = B_TRUE; entry_cnt++; /* Buffer this 'title' line */ if ((buffer = (char **)realloc(buffer, sizeof (char *)*(nlines + 1))) == NULL) { ret = BE_ERR_NOMEM; goto cleanup; } if ((buffer[nlines++] = strdup(menu_buf)) == NULL) { ret = BE_ERR_NOMEM; goto cleanup; } } else if (strcmp(tok, "bootfs") == 0) { char *bootfs = NULL; /* * Found a 'bootfs' line. See if it matches the * BE we're looking for. */ if ((bootfs = strtok(NULL, BE_WHITE_SPACE)) == NULL || strcmp(bootfs, be_root_ds) != 0) { /* * Either there's nothing after the 'bootfs' * or this is not the BE we're looking for, * write out the line(s) we've buffered since * finding the title. */ for (i = 0; i < nlines; i++) { (void) fputs(buffer[i], tmp_menu_fp); free(buffer[i]); } free(buffer); buffer = NULL; nlines = 0; /* * Turn writing back on, and turn off buffering * since this isn't the entry we're looking * for. */ write = B_TRUE; do_buffer = B_FALSE; /* Write this 'bootfs' line out. */ (void) fputs(menu_buf, tmp_menu_fp); } else { /* * Found the entry we're looking for. * Record its entry number, increment the * number of entries we've deleted, and turn * writing off. Also, throw away the lines * we've buffered for this entry so far, we * don't need them. */ entry_del = entry_cnt - 1; num_entry_del++; write = B_FALSE; do_buffer = B_FALSE; for (i = 0; i < nlines; i++) { free(buffer[i]); } free(buffer); buffer = NULL; nlines = 0; } } else { if (do_buffer) { /* Buffer this line */ if ((buffer = (char **)realloc(buffer, sizeof (char *)*(nlines + 1))) == NULL) { ret = BE_ERR_NOMEM; goto cleanup; } if ((buffer[nlines++] = strdup(menu_buf)) == NULL) { ret = BE_ERR_NOMEM; goto cleanup; } } else if (write) { /* Write this line out */ (void) fputs(menu_buf, tmp_menu_fp); } } } (void) fclose(menu_fp); menu_fp = NULL; (void) fclose(tmp_menu_fp); tmp_menu_fp = NULL; /* Copy the modified menu.lst into place */ if (rename(tmp_menu, menu) != 0) { err = errno; be_print_err(gettext("be_remove_menu: " "failed to rename file %s to %s: %s\n"), tmp_menu, menu, strerror(err)); ret = errno_to_be_err(err); goto cleanup; } free(tmp_menu); tmp_menu = NULL; /* * If we've removed an entry, see if we need to * adjust the default value in the menu.lst. If the * entry we've deleted comes before the default entry * we need to adjust the default value accordingly. * * be_has_grub is used here to check to see if this system * supports grub. */ if (be_has_grub() && num_entry_del > 0) { if (entry_del <= default_entry) { default_entry = default_entry - num_entry_del; if (default_entry < 0) default_entry = 0; /* * Adjust the default value by rewriting the * menu.lst file. This may be overkill, but to * preserve the location of the 'default' entry * in the file, we need to do this. */ /* Get handle to boot menu file */ if ((menu_fp = fopen(menu, "r")) == NULL) { err = errno; be_print_err(gettext("be_remove_menu: " "failed to open menu.lst (%s): %s\n"), menu, strerror(err)); ret = errno_to_be_err(err); goto cleanup; } /* Create a tmp file for the modified menu.lst */ tmp_menu_len = strlen(menu) + 7; if ((tmp_menu = (char *)malloc(tmp_menu_len)) == NULL) { be_print_err(gettext("be_remove_menu: " "malloc failed\n")); ret = BE_ERR_NOMEM; goto cleanup; } (void) memset(tmp_menu, 0, tmp_menu_len); (void) strlcpy(tmp_menu, menu, tmp_menu_len); (void) strlcat(tmp_menu, "XXXXXX", tmp_menu_len); if ((fd = mkstemp(tmp_menu)) == -1) { err = errno; be_print_err(gettext("be_remove_menu: " "mkstemp failed: %s\n"), strerror(err)); ret = errno_to_be_err(err); free(tmp_menu); tmp_menu = NULL; goto cleanup; } if ((tmp_menu_fp = fdopen(fd, "w")) == NULL) { err = errno; be_print_err(gettext("be_remove_menu: " "could not open tmp file for write: %s\n"), strerror(err)); (void) close(fd); ret = errno_to_be_err(err); goto cleanup; } while (fgets(menu_buf, BUFSIZ, menu_fp)) { char tline [BUFSIZ]; char *tok = NULL; (void) strlcpy(tline, menu_buf, sizeof (tline)); /* Tokenize line */ tok = strtok(tline, BE_WHITE_SPACE); if (tok == NULL) { /* Found empty line, write it out */ (void) fputs(menu_buf, tmp_menu_fp); } else if (strcmp(tok, "default") == 0) { /* Found the default line, adjust it */ (void) snprintf(tline, sizeof (tline), "default %d\n", default_entry); (void) fputs(tline, tmp_menu_fp); } else { /* Pass through all other lines */ (void) fputs(menu_buf, tmp_menu_fp); } } (void) fclose(menu_fp); menu_fp = NULL; (void) fclose(tmp_menu_fp); tmp_menu_fp = NULL; /* Copy the modified menu.lst into place */ if (rename(tmp_menu, menu) != 0) { err = errno; be_print_err(gettext("be_remove_menu: " "failed to rename file %s to %s: %s\n"), tmp_menu, menu, strerror(err)); ret = errno_to_be_err(err); goto cleanup; } free(tmp_menu); tmp_menu = NULL; } } /* Set the perms and ownership of the updated file */ if (chmod(menu, sb.st_mode) != 0) { err = errno; be_print_err(gettext("be_remove_menu: " "failed to chmod %s: %s\n"), menu, strerror(err)); ret = errno_to_be_err(err); goto cleanup; } if (chown(menu, sb.st_uid, sb.st_gid) != 0) { err = errno; be_print_err(gettext("be_remove_menu: " "failed to chown %s: %s\n"), menu, strerror(err)); ret = errno_to_be_err(err); goto cleanup; } cleanup: if (pool_mounted) { int err = BE_SUCCESS; err = be_unmount_pool(zhp, ptmp_mntpnt, orig_mntpnt); if (ret == BE_SUCCESS) ret = err; free(orig_mntpnt); free(ptmp_mntpnt); } ZFS_CLOSE(zhp); free(buffer); if (menu_fp != NULL) (void) fclose(menu_fp); if (tmp_menu_fp != NULL) (void) fclose(tmp_menu_fp); if (tmp_menu != NULL) { (void) unlink(tmp_menu); free(tmp_menu); } return (ret); } /* * Function: be_default_grub_bootfs * Description: This function returns the dataset in the default entry of * the grub menu. If no default entry is found with a valid bootfs * entry NULL is returned. * Parameters: * be_root_pool - This is the name of the root pool where the * grub menu can be found. * def_bootfs - This is used to pass back the bootfs string. On * error NULL is returned here. * Returns: * Success - BE_SUCCESS is returned. * Failure - a be_errno_t is returned. * Scope: * Semi-private (library wide use only) */ int be_default_grub_bootfs(const char *be_root_pool, char **def_bootfs) { zfs_handle_t *zhp = NULL; char grub_file[MAXPATHLEN]; FILE *menu_fp; char line[BUFSIZ]; char *pool_mntpnt = NULL; char *ptmp_mntpnt = NULL; char *orig_mntpnt = NULL; int default_entry = 0, entries = 0; int found_default = 0; int ret = BE_SUCCESS; boolean_t pool_mounted = B_FALSE; errno = 0; /* * Check to see if this system supports grub */ if (!be_has_grub()) { be_print_err(gettext("be_default_grub_bootfs: operation " "not supported on this architecture\n")); return (BE_ERR_NOTSUP); } *def_bootfs = NULL; /* Get handle to pool dataset */ if ((zhp = zfs_open(g_zfs, be_root_pool, ZFS_TYPE_DATASET)) == NULL) { be_print_err(gettext("be_default_grub_bootfs: " "failed to open pool dataset for %s: %s"), be_root_pool, libzfs_error_description(g_zfs)); return (zfs_err_to_be_err(g_zfs)); } /* * Check to see if the pool's dataset is mounted. If it isn't we'll * attempt to mount it. */ if ((ret = be_mount_pool(zhp, &ptmp_mntpnt, &orig_mntpnt, &pool_mounted)) != BE_SUCCESS) { be_print_err(gettext("be_default_grub_bootfs: pool dataset " "(%s) could not be mounted\n"), be_root_pool); ZFS_CLOSE(zhp); return (ret); } /* * Get the mountpoint for the root pool dataset. */ if (!zfs_is_mounted(zhp, &pool_mntpnt)) { be_print_err(gettext("be_default_grub_bootfs: failed " "to get mount point for the root pool. Can't set " "the default BE in the grub menu.\n")); ret = BE_ERR_NO_MENU; goto cleanup; } (void) snprintf(grub_file, MAXPATHLEN, "%s%s", pool_mntpnt, BE_GRUB_MENU); if ((ret = be_open_menu((char *)be_root_pool, grub_file, &menu_fp, "r", B_FALSE)) != BE_SUCCESS) { goto cleanup; } else if (menu_fp == NULL) { ret = BE_ERR_NO_MENU; goto cleanup; } free(pool_mntpnt); pool_mntpnt = NULL; while (fgets(line, BUFSIZ, menu_fp)) { char *tok = strtok(line, BE_WHITE_SPACE); if (tok != NULL && tok[0] != '#') { if (!found_default) { if (strcmp(tok, "default") == 0) { tok = strtok(NULL, BE_WHITE_SPACE); if (tok != NULL) { default_entry = atoi(tok); rewind(menu_fp); found_default = 1; } } continue; } if (strcmp(tok, "title") == 0) { entries++; } else if (default_entry == entries - 1) { if (strcmp(tok, "bootfs") == 0) { tok = strtok(NULL, BE_WHITE_SPACE); (void) fclose(menu_fp); if (tok == NULL) { ret = BE_SUCCESS; goto cleanup; } if ((*def_bootfs = strdup(tok)) != NULL) { ret = BE_SUCCESS; goto cleanup; } be_print_err(gettext( "be_default_grub_bootfs: " "memory allocation failed\n")); ret = BE_ERR_NOMEM; goto cleanup; } } else if (default_entry < entries - 1) { /* * no bootfs entry for the default entry. */ break; } } } (void) fclose(menu_fp); cleanup: if (pool_mounted) { int err = BE_SUCCESS; err = be_unmount_pool(zhp, ptmp_mntpnt, orig_mntpnt); if (ret == BE_SUCCESS) ret = err; free(orig_mntpnt); free(ptmp_mntpnt); } ZFS_CLOSE(zhp); return (ret); } /* * Function: be_change_grub_default * Description: This function takes two parameters. These are the name of * the BE we want to have as the default booted in the grub * menu and the root pool where the path to the grub menu exists. * The code takes this and finds the BE's entry in the grub menu * and changes the default entry to point to that entry in the * list. * Parameters: * be_name - This is the name of the BE wanted as the default * for the next boot. * be_root_pool - This is the name of the root pool where the * grub menu can be found. * Returns: * BE_SUCCESS - Success * be_errno_t - Failure * Scope: * Semi-private (library wide use only) */ int be_change_grub_default(char *be_name, char *be_root_pool) { zfs_handle_t *zhp = NULL; char grub_file[MAXPATHLEN]; char *temp_grub = NULL; char *pool_mntpnt = NULL; char *ptmp_mntpnt = NULL; char *orig_mntpnt = NULL; char line[BUFSIZ]; char temp_line[BUFSIZ]; char be_root_ds[MAXPATHLEN]; FILE *grub_fp = NULL; FILE *temp_fp = NULL; struct stat sb; int temp_grub_len = 0; int fd, entries = 0; int err = 0; int ret = BE_SUCCESS; boolean_t found_default = B_FALSE; boolean_t pool_mounted = B_FALSE; errno = 0; /* * Check to see if this system supports grub */ if (!be_has_grub()) { be_print_err(gettext("be_change_grub_default: operation " "not supported on this architecture\n")); return (BE_ERR_NOTSUP); } /* Generate string for BE's root dataset */ be_make_root_ds(be_root_pool, be_name, be_root_ds, sizeof (be_root_ds)); /* Get handle to pool dataset */ if ((zhp = zfs_open(g_zfs, be_root_pool, ZFS_TYPE_DATASET)) == NULL) { be_print_err(gettext("be_change_grub_default: " "failed to open pool dataset for %s: %s"), be_root_pool, libzfs_error_description(g_zfs)); return (zfs_err_to_be_err(g_zfs)); } /* * Check to see if the pool's dataset is mounted. If it isn't we'll * attempt to mount it. */ if ((ret = be_mount_pool(zhp, &ptmp_mntpnt, &orig_mntpnt, &pool_mounted)) != BE_SUCCESS) { be_print_err(gettext("be_change_grub_default: pool dataset " "(%s) could not be mounted\n"), be_root_pool); ZFS_CLOSE(zhp); return (ret); } /* * Get the mountpoint for the root pool dataset. */ if (!zfs_is_mounted(zhp, &pool_mntpnt)) { be_print_err(gettext("be_change_grub_default: pool " "dataset (%s) is not mounted. Can't set " "the default BE in the grub menu.\n"), be_root_pool); ret = BE_ERR_NO_MENU; goto cleanup; } (void) snprintf(grub_file, MAXPATHLEN, "%s%s", pool_mntpnt, BE_GRUB_MENU); if ((ret = be_open_menu(be_root_pool, grub_file, &grub_fp, "r+", B_TRUE)) != BE_SUCCESS) { goto cleanup; } else if (grub_fp == NULL) { ret = BE_ERR_NO_MENU; goto cleanup; } free(pool_mntpnt); pool_mntpnt = NULL; /* Grab the stats of the original menu file */ if (stat(grub_file, &sb) != 0) { err = errno; be_print_err(gettext("be_change_grub_default: " "failed to stat file %s: %s\n"), grub_file, strerror(err)); ret = errno_to_be_err(err); goto cleanup; } /* Create a tmp file for the modified menu.lst */ temp_grub_len = strlen(grub_file) + 7; if ((temp_grub = (char *)malloc(temp_grub_len)) == NULL) { be_print_err(gettext("be_change_grub_default: " "malloc failed\n")); ret = BE_ERR_NOMEM; goto cleanup; } (void) memset(temp_grub, 0, temp_grub_len); (void) strlcpy(temp_grub, grub_file, temp_grub_len); (void) strlcat(temp_grub, "XXXXXX", temp_grub_len); if ((fd = mkstemp(temp_grub)) == -1) { err = errno; be_print_err(gettext("be_change_grub_default: " "mkstemp failed: %s\n"), strerror(err)); ret = errno_to_be_err(err); free(temp_grub); temp_grub = NULL; goto cleanup; } if ((temp_fp = fdopen(fd, "w")) == NULL) { err = errno; be_print_err(gettext("be_change_grub_default: " "failed to open %s file: %s\n"), temp_grub, strerror(err)); (void) close(fd); ret = errno_to_be_err(err); goto cleanup; } while (fgets(line, BUFSIZ, grub_fp)) { char *tok = strtok(line, BE_WHITE_SPACE); if (tok == NULL || tok[0] == '#') { continue; } else if (strcmp(tok, "title") == 0) { entries++; continue; } else if (strcmp(tok, "bootfs") == 0) { char *bootfs = strtok(NULL, BE_WHITE_SPACE); if (bootfs == NULL) continue; if (strcmp(bootfs, be_root_ds) == 0) { found_default = B_TRUE; break; } } } if (!found_default) { be_print_err(gettext("be_change_grub_default: failed " "to find entry for %s in the grub menu\n"), be_name); ret = BE_ERR_BE_NOENT; goto cleanup; } rewind(grub_fp); while (fgets(line, BUFSIZ, grub_fp)) { char *tok = NULL; (void) strncpy(temp_line, line, BUFSIZ); if ((tok = strtok(temp_line, BE_WHITE_SPACE)) != NULL && strcmp(tok, "default") == 0) { (void) snprintf(temp_line, BUFSIZ, "default %d\n", entries - 1 >= 0 ? entries - 1 : 0); (void) fputs(temp_line, temp_fp); } else { (void) fputs(line, temp_fp); } } (void) fclose(grub_fp); grub_fp = NULL; (void) fclose(temp_fp); temp_fp = NULL; if (rename(temp_grub, grub_file) != 0) { err = errno; be_print_err(gettext("be_change_grub_default: " "failed to rename file %s to %s: %s\n"), temp_grub, grub_file, strerror(err)); ret = errno_to_be_err(err); goto cleanup; } free(temp_grub); temp_grub = NULL; /* Set the perms and ownership of the updated file */ if (chmod(grub_file, sb.st_mode) != 0) { err = errno; be_print_err(gettext("be_change_grub_default: " "failed to chmod %s: %s\n"), grub_file, strerror(err)); ret = errno_to_be_err(err); goto cleanup; } if (chown(grub_file, sb.st_uid, sb.st_gid) != 0) { err = errno; be_print_err(gettext("be_change_grub_default: " "failed to chown %s: %s\n"), grub_file, strerror(err)); ret = errno_to_be_err(err); } cleanup: if (pool_mounted) { int err = BE_SUCCESS; err = be_unmount_pool(zhp, ptmp_mntpnt, orig_mntpnt); if (ret == BE_SUCCESS) ret = err; free(orig_mntpnt); free(ptmp_mntpnt); } ZFS_CLOSE(zhp); if (grub_fp != NULL) (void) fclose(grub_fp); if (temp_fp != NULL) (void) fclose(temp_fp); if (temp_grub != NULL) { (void) unlink(temp_grub); free(temp_grub); } return (ret); } /* * Function: be_update_menu * Description: This function is used by be_rename to change the BE name in * an existing entry in the grub menu to the new name of the BE. * Parameters: * be_orig_name - the original name of the BE * be_new_name - the new name the BE is being renameed to. * be_root_pool - The pool which contains the grub menu * boot_pool - the pool where the BE is, if different than * the pool containing the boot menu. If this is * NULL it will be set to be_root_pool. * Returns: * BE_SUCCESS - Success * be_errno_t - Failure * Scope: * Semi-private (library wide use only) */ int be_update_menu(char *be_orig_name, char *be_new_name, char *be_root_pool, char *boot_pool) { zfs_handle_t *zhp = NULL; char menu_file[MAXPATHLEN]; char be_root_ds[MAXPATHLEN]; char be_new_root_ds[MAXPATHLEN]; char line[BUFSIZ]; char *pool_mntpnt = NULL; char *ptmp_mntpnt = NULL; char *orig_mntpnt = NULL; char *temp_menu = NULL; FILE *menu_fp = NULL; FILE *new_fp = NULL; struct stat sb; int temp_menu_len = 0; int tmp_fd; int ret = BE_SUCCESS; int err = 0; boolean_t pool_mounted = B_FALSE; errno = 0; if (boot_pool == NULL) boot_pool = be_root_pool; if ((zhp = zfs_open(g_zfs, be_root_pool, ZFS_TYPE_DATASET)) == NULL) { be_print_err(gettext("be_update_menu: failed to open " "pool dataset for %s: %s\n"), be_root_pool, libzfs_error_description(g_zfs)); return (zfs_err_to_be_err(g_zfs)); } /* * Check to see if the pool's dataset is mounted. If it isn't we'll * attempt to mount it. */ if ((ret = be_mount_pool(zhp, &ptmp_mntpnt, &orig_mntpnt, &pool_mounted)) != BE_SUCCESS) { be_print_err(gettext("be_update_menu: pool dataset " "(%s) could not be mounted\n"), be_root_pool); ZFS_CLOSE(zhp); return (ret); } /* * Get the mountpoint for the root pool dataset. */ if (!zfs_is_mounted(zhp, &pool_mntpnt)) { be_print_err(gettext("be_update_menu: failed " "to get mount point for the root pool. Can't set " "the default BE in the grub menu.\n")); ret = BE_ERR_NO_MENU; goto cleanup; } /* * Check to see if this system supports grub */ if (be_has_grub()) { (void) snprintf(menu_file, sizeof (menu_file), "%s%s", pool_mntpnt, BE_GRUB_MENU); } else { (void) snprintf(menu_file, sizeof (menu_file), "%s%s", pool_mntpnt, BE_SPARC_MENU); } be_make_root_ds(be_root_pool, be_orig_name, be_root_ds, sizeof (be_root_ds)); be_make_root_ds(be_root_pool, be_new_name, be_new_root_ds, sizeof (be_new_root_ds)); if ((ret = be_open_menu(be_root_pool, menu_file, &menu_fp, "r", B_TRUE)) != BE_SUCCESS) { goto cleanup; } else if (menu_fp == NULL) { ret = BE_ERR_NO_MENU; goto cleanup; } free(pool_mntpnt); pool_mntpnt = NULL; /* Grab the stat of the original menu file */ if (stat(menu_file, &sb) != 0) { err = errno; be_print_err(gettext("be_update_menu: " "failed to stat file %s: %s\n"), menu_file, strerror(err)); (void) fclose(menu_fp); ret = errno_to_be_err(err); goto cleanup; } /* Create tmp file for modified menu.lst */ temp_menu_len = strlen(menu_file) + 7; if ((temp_menu = (char *)malloc(temp_menu_len)) == NULL) { be_print_err(gettext("be_update_menu: " "malloc failed\n")); (void) fclose(menu_fp); ret = BE_ERR_NOMEM; goto cleanup; } (void) memset(temp_menu, 0, temp_menu_len); (void) strlcpy(temp_menu, menu_file, temp_menu_len); (void) strlcat(temp_menu, "XXXXXX", temp_menu_len); if ((tmp_fd = mkstemp(temp_menu)) == -1) { err = errno; be_print_err(gettext("be_update_menu: " "mkstemp failed: %s\n"), strerror(err)); (void) fclose(menu_fp); free(temp_menu); ret = errno_to_be_err(err); goto cleanup; } if ((new_fp = fdopen(tmp_fd, "w")) == NULL) { err = errno; be_print_err(gettext("be_update_menu: " "fdopen failed: %s\n"), strerror(err)); (void) close(tmp_fd); (void) fclose(menu_fp); free(temp_menu); ret = errno_to_be_err(err); goto cleanup; } while (fgets(line, BUFSIZ, menu_fp)) { char tline[BUFSIZ]; char new_line[BUFSIZ]; char *c = NULL; (void) strlcpy(tline, line, sizeof (tline)); /* Tokenize line */ c = strtok(tline, BE_WHITE_SPACE); if (c == NULL) { /* Found empty line, write it out. */ (void) fputs(line, new_fp); } else if (c[0] == '#') { /* Found a comment line, write it out. */ (void) fputs(line, new_fp); } else if (strcmp(c, "title") == 0) { char *name = NULL; char *desc = NULL; /* * Found a 'title' line, parse out BE name or * the description. */ name = strtok(NULL, BE_WHITE_SPACE); if (name == NULL) { /* * Nothing after 'title', just push * this line through */ (void) fputs(line, new_fp); } else { /* * Grab the remainder of the title which * could be a multi worded description */ desc = strtok(NULL, "\n"); if (strcmp(name, be_orig_name) == 0) { /* * The first token of the title is * the old BE name, replace it with * the new one, and write it out * along with the remainder of * description if there is one. */ if (desc) { (void) snprintf(new_line, sizeof (new_line), "title %s %s\n", be_new_name, desc); } else { (void) snprintf(new_line, sizeof (new_line), "title %s\n", be_new_name); } (void) fputs(new_line, new_fp); } else { (void) fputs(line, new_fp); } } } else if (strcmp(c, "bootfs") == 0) { /* * Found a 'bootfs' line, parse out the BE root * dataset value. */ char *root_ds = strtok(NULL, BE_WHITE_SPACE); if (root_ds == NULL) { /* * Nothing after 'bootfs', just push * this line through */ (void) fputs(line, new_fp); } else { /* * If this bootfs is the one we're renaming, * write out the new root dataset value */ if (strcmp(root_ds, be_root_ds) == 0) { (void) snprintf(new_line, sizeof (new_line), "bootfs %s\n", be_new_root_ds); (void) fputs(new_line, new_fp); } else { (void) fputs(line, new_fp); } } } else { /* * Found some other line we don't care * about, write it out. */ (void) fputs(line, new_fp); } } (void) fclose(menu_fp); (void) fclose(new_fp); (void) close(tmp_fd); if (rename(temp_menu, menu_file) != 0) { err = errno; be_print_err(gettext("be_update_menu: " "failed to rename file %s to %s: %s\n"), temp_menu, menu_file, strerror(err)); ret = errno_to_be_err(err); } free(temp_menu); /* Set the perms and ownership of the updated file */ if (chmod(menu_file, sb.st_mode) != 0) { err = errno; be_print_err(gettext("be_update_menu: " "failed to chmod %s: %s\n"), menu_file, strerror(err)); ret = errno_to_be_err(err); goto cleanup; } if (chown(menu_file, sb.st_uid, sb.st_gid) != 0) { err = errno; be_print_err(gettext("be_update_menu: " "failed to chown %s: %s\n"), menu_file, strerror(err)); ret = errno_to_be_err(err); } cleanup: if (pool_mounted) { int err = BE_SUCCESS; err = be_unmount_pool(zhp, ptmp_mntpnt, orig_mntpnt); if (ret == BE_SUCCESS) ret = err; free(orig_mntpnt); free(ptmp_mntpnt); } ZFS_CLOSE(zhp); return (ret); } /* * Function: be_has_menu_entry * Description: Checks to see if the BEs root dataset has an entry in the grub * menu. * Parameters: * be_dataset - The root dataset of the BE * be_root_pool - The pool which contains the boot menu * entry - A pointer the the entry number of the BE if found. * Returns: * B_TRUE - Success * B_FALSE - Failure * Scope: * Semi-private (library wide use only) */ boolean_t be_has_menu_entry(char *be_dataset, char *be_root_pool, int *entry) { zfs_handle_t *zhp = NULL; char menu_file[MAXPATHLEN]; FILE *menu_fp; char line[BUFSIZ]; char *last; char *rpool_mntpnt = NULL; char *ptmp_mntpnt = NULL; char *orig_mntpnt = NULL; int ent_num = 0; boolean_t ret = 0; boolean_t pool_mounted = B_FALSE; /* * Check to see if this system supports grub */ if ((zhp = zfs_open(g_zfs, be_root_pool, ZFS_TYPE_DATASET)) == NULL) { be_print_err(gettext("be_has_menu_entry: failed to open " "pool dataset for %s: %s\n"), be_root_pool, libzfs_error_description(g_zfs)); return (B_FALSE); } /* * Check to see if the pool's dataset is mounted. If it isn't we'll * attempt to mount it. */ if (be_mount_pool(zhp, &ptmp_mntpnt, &orig_mntpnt, &pool_mounted) != 0) { be_print_err(gettext("be_has_menu_entry: pool dataset " "(%s) could not be mounted\n"), be_root_pool); ZFS_CLOSE(zhp); return (B_FALSE); } /* * Get the mountpoint for the root pool dataset. */ if (!zfs_is_mounted(zhp, &rpool_mntpnt)) { be_print_err(gettext("be_has_menu_entry: pool " "dataset (%s) is not mounted. Can't set " "the default BE in the grub menu.\n"), be_root_pool); ret = B_FALSE; goto cleanup; } if (be_has_grub()) { (void) snprintf(menu_file, MAXPATHLEN, "/%s%s", rpool_mntpnt, BE_GRUB_MENU); } else { (void) snprintf(menu_file, MAXPATHLEN, "/%s%s", rpool_mntpnt, BE_SPARC_MENU); } if (be_open_menu(be_root_pool, menu_file, &menu_fp, "r", B_FALSE) != 0) { ret = B_FALSE; goto cleanup; } else if (menu_fp == NULL) { ret = B_FALSE; goto cleanup; } free(rpool_mntpnt); rpool_mntpnt = NULL; while (fgets(line, BUFSIZ, menu_fp)) { char *tok = strtok_r(line, BE_WHITE_SPACE, &last); if (tok != NULL && tok[0] != '#') { if (strcmp(tok, "bootfs") == 0) { tok = strtok_r(last, BE_WHITE_SPACE, &last); if (tok != NULL && strcmp(tok, be_dataset) == 0) { (void) fclose(menu_fp); /* * The entry number needs to be * decremented here because the title * will always be the first line for * an entry. Because of this we'll * always be off by one entry when we * check for bootfs. */ *entry = ent_num - 1; ret = B_TRUE; goto cleanup; } } else if (strcmp(tok, "title") == 0) ent_num++; } } cleanup: if (pool_mounted) { (void) be_unmount_pool(zhp, ptmp_mntpnt, orig_mntpnt); free(orig_mntpnt); free(ptmp_mntpnt); } ZFS_CLOSE(zhp); (void) fclose(menu_fp); return (ret); } /* * Function: be_update_vfstab * Description: This function digs into a BE's vfstab and updates all * entries with file systems listed in be_fs_list_data_t. * The entry's root container dataset and be_name will be * updated with the parameters passed in. * Parameters: * be_name - name of BE to update * old_rc_loc - dataset under which the root container dataset * of the old BE resides in. * new_rc_loc - dataset under which the root container dataset * of the new BE resides in. * fld - be_fs_list_data_t pointer providing the list of * file systems to look for in vfstab. * mountpoint - directory of where BE is currently mounted. * If NULL, then BE is not currently mounted. * Returns: * BE_SUCCESS - Success * be_errno_t - Failure * Scope: * Semi-private (library wide use only) */ int be_update_vfstab(char *be_name, char *old_rc_loc, char *new_rc_loc, be_fs_list_data_t *fld, char *mountpoint) { char *tmp_mountpoint = NULL; char alt_vfstab[MAXPATHLEN]; int ret = BE_SUCCESS, err = BE_SUCCESS; if (fld == NULL || fld->fs_list == NULL || fld->fs_num == 0) return (BE_SUCCESS); /* If BE not already mounted, mount the BE */ if (mountpoint == NULL) { if ((ret = _be_mount(be_name, &tmp_mountpoint, BE_MOUNT_FLAG_NO_ZONES)) != BE_SUCCESS) { be_print_err(gettext("be_update_vfstab: " "failed to mount BE (%s)\n"), be_name); return (ret); } } else { tmp_mountpoint = mountpoint; } /* Get string for vfstab in the mounted BE. */ (void) snprintf(alt_vfstab, sizeof (alt_vfstab), "%s/etc/vfstab", tmp_mountpoint); /* Update the vfstab */ ret = _update_vfstab(alt_vfstab, be_name, old_rc_loc, new_rc_loc, fld); /* Unmount BE if we mounted it */ if (mountpoint == NULL) { if ((err = _be_unmount(be_name, 0)) == BE_SUCCESS) { /* Remove temporary mountpoint */ (void) rmdir(tmp_mountpoint); } else { be_print_err(gettext("be_update_vfstab: " "failed to unmount BE %s mounted at %s\n"), be_name, tmp_mountpoint); if (ret == BE_SUCCESS) ret = err; } free(tmp_mountpoint); } return (ret); } /* * Function: be_update_zone_vfstab * Description: This function digs into a zone BE's vfstab and updates all * entries with file systems listed in be_fs_list_data_t. * The entry's root container dataset and be_name will be * updated with the parameters passed in. * Parameters: * zhp - zfs_handle_t pointer to zone root dataset. * be_name - name of zone BE to update * old_rc_loc - dataset under which the root container dataset * of the old zone BE resides in. * new_rc_loc - dataset under which the root container dataset * of the new zone BE resides in. * fld - be_fs_list_data_t pointer providing the list of * file systems to look for in vfstab. * Returns: * BE_SUCCESS - Success * be_errno_t - Failure * Scope: * Semi-private (library wide use only) */ int be_update_zone_vfstab(zfs_handle_t *zhp, char *be_name, char *old_rc_loc, char *new_rc_loc, be_fs_list_data_t *fld) { be_mount_data_t md = { 0 }; be_unmount_data_t ud = { 0 }; char alt_vfstab[MAXPATHLEN]; boolean_t mounted_here = B_FALSE; int ret = BE_SUCCESS; /* * If zone root not already mounted, mount it at a * temporary location. */ if (!zfs_is_mounted(zhp, &md.altroot)) { /* Generate temporary mountpoint to mount zone root */ if ((ret = be_make_tmp_mountpoint(&md.altroot)) != BE_SUCCESS) { be_print_err(gettext("be_update_zone_vfstab: " "failed to make temporary mountpoint to " "mount zone root\n")); return (ret); } if (be_mount_zone_root(zhp, &md) != BE_SUCCESS) { be_print_err(gettext("be_update_zone_vfstab: " "failed to mount zone root %s\n"), zfs_get_name(zhp)); free(md.altroot); return (BE_ERR_MOUNT_ZONEROOT); } mounted_here = B_TRUE; } /* Get string from vfstab in the mounted zone BE */ (void) snprintf(alt_vfstab, sizeof (alt_vfstab), "%s/etc/vfstab", md.altroot); /* Update the vfstab */ ret = _update_vfstab(alt_vfstab, be_name, old_rc_loc, new_rc_loc, fld); /* Unmount zone root if we mounted it */ if (mounted_here) { ud.force = B_TRUE; if (be_unmount_zone_root(zhp, &ud) == BE_SUCCESS) { /* Remove the temporary mountpoint */ (void) rmdir(md.altroot); } else { be_print_err(gettext("be_update_zone_vfstab: " "failed to unmount zone root %s from %s\n"), zfs_get_name(zhp), md.altroot); if (ret == 0) ret = BE_ERR_UMOUNT_ZONEROOT; } } free(md.altroot); return (ret); } /* * Function: be_auto_snap_name * Description: Generate an auto snapshot name constructed based on the * current date and time. The auto snapshot name is of the form: * * -