/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License, Version 1.0 only * (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 2003 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ #pragma ident "%Z%%M% %I% %E% SMI" /* * Traverses /etc/mnttab in order to find mounted file systems. */ #include <errno.h> #include <unistd.h> #include <stdlib.h> #include <stdio.h> #include <sys/mnttab.h> #include <sys/types.h> #include <sys/statvfs.h> #include <strings.h> #include "libfsmgt.h" /* * Private variables */ /* * Private method declarations */ static fs_mntlist_t *create_mntlist_entry(struct mnttab mnttab_entry); static fs_mntlist_t *create_extmntlist_entry(struct extmnttab mnttab_entry); static struct mnttab *create_mnttab_filter(char *resource, char *mountp, char *fstype, char *mntopts, char *time); static void find_overlayed_filesystems(fs_mntlist_t *mnt_list, boolean_t filtered_list, int *errp); static void free_mnttab_entry(struct mnttab *mnttab_entry); static char *is_option(char *opt_string, char *opt, int *errp); boolean_t is_overlayed(fs_mntlist_t *complete_mnt_list, char *mountp); /* * Public methods */ void fs_free_mount_list(fs_mntlist_t *headp) { fs_mntlist_t *tmp; while (headp != NULL) { tmp = headp->next; free(headp->resource); free(headp->mountp); free(headp->fstype); free(headp->mntopts); free(headp->time); headp->next = NULL; free(headp); headp = tmp; } } /* fs_free_mount_list */ unsigned long long fs_get_availablesize(char *mntpnt, int *errp) { struct statvfs64 stvfs; unsigned long long availablesize; *errp = 0; if (mntpnt == NULL) { /* * Set errp to invalid parameter - EINVAL */ *errp = EINVAL; return (0); } if (statvfs64(mntpnt, &stvfs) != -1) { availablesize = stvfs.f_bfree; availablesize = availablesize * stvfs.f_frsize; } else { *errp = errno; return (0); } /* if (statvfs64(mntpnt, &stvfs) != -1) */ return (availablesize); } /* fs_get_availablesize */ unsigned long long fs_get_avail_for_nonsuperuser_size(char *mntpnt, int *errp) { struct statvfs64 stvfs; unsigned long long avail_for_nonsu_size; *errp = 0; if (mntpnt == NULL) { /* * Set errp to invalid parameter - EINVAL */ *errp = EINVAL; return (0); } if (statvfs64(mntpnt, &stvfs) != -1) { avail_for_nonsu_size = stvfs.f_bavail; avail_for_nonsu_size = avail_for_nonsu_size * stvfs.f_frsize; } else { *errp = errno; return (0); } /* if (statvfs64(mntpnt, &stvfs) != -1) */ return (avail_for_nonsu_size); } /* fs_get_avail_for_nonsuperuser_size(char *mntpnt, int *errp) */ unsigned long long fs_get_blocksize(char *mntpnt, int *errp) { struct statvfs64 stvfs; unsigned long long blocksize; *errp = 0; if (mntpnt == NULL) { /* * Set errp to invalid parameter - EINVAL */ *errp = EINVAL; return (0); } if (statvfs64(mntpnt, &stvfs) != -1) { blocksize = stvfs.f_bsize; } else { *errp = errno; return (0); } /* if (statvfs64(mntpnt, &stvfs) != -1) */ return (blocksize); } /* fs_get_blocksize */ fs_mntlist_t * fs_get_filtered_mount_list(char *resource, char *mountp, char *fstype, char *mntopts, char *time, boolean_t find_overlays, int *errp) { fs_mntlist_t *newp; fs_mntlist_t *headp; fs_mntlist_t *tailp; FILE *fp; *errp = 0; headp = NULL; tailp = NULL; if ((fp = fopen(MNTTAB, "r")) != NULL) { struct mnttab mnttab_entry; struct mnttab *search_entry; search_entry = create_mnttab_filter(resource, mountp, fstype, mntopts, time); if (search_entry == NULL) { /* * Out of memory */ fs_free_mount_list(headp); (void) fclose(fp); *errp = ENOMEM; return (NULL); } while (getmntany(fp, &mnttab_entry, search_entry) == 0) { /* Add to list to be returned */ newp = create_mntlist_entry(mnttab_entry); if (newp == NULL) { /* * Out of memory */ fs_free_mount_list(headp); (void) fclose(fp); *errp = ENOMEM; return (NULL); } if (headp == NULL) { headp = newp; tailp = newp; } else { tailp->next = newp; tailp = newp; } } free_mnttab_entry(search_entry); (void) fclose(fp); if (find_overlays == B_TRUE) find_overlayed_filesystems(headp, B_TRUE, errp); } else { *errp = errno; } /* if ((fp = fopen(MNTTAB, "r")) != NULL) */ return (headp); } /* fs_get_filtered_mount_list */ unsigned long fs_get_fragsize(char *mntpnt, int *errp) { struct statvfs64 stvfs; unsigned long fragsize; *errp = 0; if (mntpnt == NULL) { /* * Set errp to invalid parameter - EINVAL */ *errp = EINVAL; return (0); } if (statvfs64(mntpnt, &stvfs) != -1) { fragsize = stvfs.f_frsize; } else { *errp = errno; return (0); } /* (statvfs64(mntpnt, &stvfs) != -1) */ return (fragsize); } /* fs_get_fragsize(char *mntpnt, int *errp) */ unsigned long fs_get_maxfilenamelen(char *mntpnt, int *errp) { long int returned_val; unsigned long maxfilenamelen; *errp = 0; if (mntpnt == NULL) { /* * Set errp to invalid parameter - EINVAL */ *errp = EINVAL; return (0); } returned_val = pathconf(mntpnt, _PC_PATH_MAX); if (returned_val != -1) { maxfilenamelen = (unsigned long)returned_val; } else { *errp = errno; return (0); } return (maxfilenamelen); } /* fs_get_maxfilenamelen */ fs_mntlist_t * fs_get_mounts_by_mntopt(char *mntopt, boolean_t find_overlays, int *errp) { fs_mntlist_t *newp; fs_mntlist_t *headp; fs_mntlist_t *tailp; FILE *fp; *errp = 0; headp = NULL; tailp = NULL; if (mntopt == NULL) return (NULL); if ((fp = fopen(MNTTAB, "r")) != NULL) { struct mnttab mnttab_entry; char *opt_found; while (getmntent(fp, &mnttab_entry) == 0) { opt_found = hasmntopt(&mnttab_entry, mntopt); if (opt_found != NULL) { /* * Add to list to be returned */ newp = create_mntlist_entry(mnttab_entry); if (newp == NULL) { /* * Out of memory */ fs_free_mount_list(headp); (void) fclose(fp); *errp = ENOMEM; return (NULL); } if (headp == NULL) { headp = newp; tailp = newp; } else { tailp->next = newp; tailp = newp; } } /* if (char != NULL) */ } (void) fclose(fp); if (find_overlays == B_TRUE) find_overlayed_filesystems(headp, B_TRUE, errp); } else { *errp = errno; } /* if ((fp = fopen(MNTTAB, "r")) != NULL) */ return (headp); } /* fs_get_mounts_by_mntpnt */ fs_mntlist_t * fs_get_mount_list(boolean_t find_overlays, int *errp) { FILE *fp; fs_mntlist_t *headp; fs_mntlist_t *tailp; fs_mntlist_t *newp; *errp = 0; headp = NULL; tailp = NULL; if ((fp = fopen(MNTTAB, "r")) != NULL) { struct extmnttab mnttab_entry; resetmnttab(fp); /* * getextmntent() Is used here so that we can use mnt_major * and mnt_minor to get the fsid. The fsid is used when * getting mount information from kstat. */ while (getextmntent(fp, &mnttab_entry, sizeof (struct extmnttab)) == 0) { newp = create_extmntlist_entry(mnttab_entry); if (newp == NULL) { /* * Out of memory */ fs_free_mount_list(headp); (void) fclose(fp); *errp = ENOMEM; return (NULL); } if (headp == NULL) { headp = newp; tailp = newp; } else { tailp->next = newp; tailp = newp; } } /* while (getmntent(fp, &mnttab_entry) == 0) */ (void) fclose(fp); if (find_overlays) find_overlayed_filesystems(headp, B_FALSE, errp); } else { *errp = errno; } /* if ((fp = fopen(MNTTAB, "r")) != NULL) */ /* * Caller must free the mount list */ return (headp); } /* fs_get_mount_list */ boolean_t fs_is_readonly(char *mntpnt, int *errp) { struct statvfs64 stvfs; boolean_t readonly; *errp = 0; if (mntpnt == NULL) { /* * Set errp to invalid parameter - EINVAL */ *errp = EINVAL; return (B_FALSE); } if (statvfs64(mntpnt, &stvfs) != -1) { readonly = stvfs.f_flag & ST_RDONLY; } else { *errp = errno; return (B_FALSE); } return (readonly); } /* fs_is_readonly */ /* * This method will parse the given comma delimited option list (optlist) for * the option passed into the function. If the option (opt) to search for * is one that sets a value such as onerror=, the value to the right of the "=" * character will be returned from the function. This function expects the * opt parameter to have the "=" character appended when searching for options * which set a value. * * If the option is found in the given optlist, the function will return the * option as found in the option list. * If the option is not found in the given optlist, the function will return * NULL. * If an error occurs, the function will return NULL and the errp will * reflect the error that has occurred. * * NOTE: The caller must free the space allocated for the return value by using * free(). */ char * fs_parse_optlist_for_option(char *optlist, char *opt, int *errp) { const char *delimiter = ","; char *token; char *return_value; char *optlist_copy; *errp = 0; optlist_copy = strdup(optlist); if (optlist_copy == NULL) { *errp = errno; return (NULL); } token = strtok(optlist_copy, delimiter); /* * Check to see if we have found the option. */ if (token == NULL) { free(optlist_copy); return (NULL); } else if ((return_value = is_option(token, opt, errp)) != NULL) { free(optlist_copy); return (return_value); } while (token != NULL) { token = NULL; token = strtok(NULL, delimiter); /* * If token is NULL then then we are at the end of the list * and we can return NULL because the option was never found in * the option list. */ if (token == NULL) { free(optlist_copy); return (NULL); } else if ((return_value = is_option(token, opt, errp)) != NULL) { free(optlist_copy); return (return_value); } } free(optlist_copy); return (NULL); } unsigned long long fs_get_totalsize(char *mntpnt, int *errp) { struct statvfs64 stvfs; unsigned long long totalsize; *errp = 0; if (mntpnt == NULL) { /* * Set errp to invalid parameter - EINVAL */ *errp = EINVAL; return (0); } if (statvfs64(mntpnt, &stvfs) != -1) { totalsize = stvfs.f_blocks; totalsize = totalsize * stvfs.f_frsize; } else { *errp = errno; return (0); } /* if (statvfs64(mntpnt, &stvfs) != -1) */ return (totalsize); } /* fs_get_totalsize */ unsigned long long fs_get_usedsize(char *mntpnt, int *errp) { struct statvfs64 stvfs; unsigned long long usedsize; *errp = 0; if (mntpnt == NULL) { /* * Set errp to invalid parameter - EINVAL */ *errp = EINVAL; return (0); } if (statvfs64(mntpnt, &stvfs) != -1) { usedsize = stvfs.f_blocks - stvfs.f_bfree; usedsize = usedsize * stvfs.f_frsize; } else { *errp = errno; return (0); } /* if (statvfs64(mntpnt, &stvfs) != -1) */ return (usedsize); } /* fs_get_usedsize */ /* * Private methods */ static fs_mntlist_t * create_mntlist_entry(struct mnttab mnttab_entry) { fs_mntlist_t *newp; newp = (fs_mntlist_t *)calloc((size_t)1, (size_t)sizeof (fs_mntlist_t)); if (newp == NULL) { /* * Out of memory */ return (NULL); } newp->resource = strdup(mnttab_entry.mnt_special); if (newp->resource == NULL) { /* * Out of memory */ fs_free_mount_list(newp); return (NULL); } newp->mountp = strdup(mnttab_entry.mnt_mountp); if (newp->mountp == NULL) { /* * Out of memory */ fs_free_mount_list(newp); return (NULL); } newp->fstype = strdup(mnttab_entry.mnt_fstype); if (newp->fstype == NULL) { /* * Out of memory */ fs_free_mount_list(newp); return (NULL); } newp->mntopts = strdup(mnttab_entry.mnt_mntopts); if (newp->mntopts == NULL) { /* * Out of memory */ fs_free_mount_list(newp); return (NULL); } newp->time = strdup(mnttab_entry.mnt_time); if (newp->time == NULL) { /* * Out of memory */ fs_free_mount_list(newp); return (NULL); } newp->next = NULL; return (newp); } /* create_mntlist_entry */ static fs_mntlist_t * create_extmntlist_entry(struct extmnttab mnttab_entry) { fs_mntlist_t *newp; newp = (fs_mntlist_t *)calloc((size_t)1, (size_t)sizeof (fs_mntlist_t)); if (newp == NULL) { /* * Out of memory */ return (NULL); } newp->resource = strdup(mnttab_entry.mnt_special); if (newp->resource == NULL) { /* * Out of memory */ fs_free_mount_list(newp); return (NULL); } newp->mountp = strdup(mnttab_entry.mnt_mountp); if (newp->mountp == NULL) { /* * Out of memory */ fs_free_mount_list(newp); return (NULL); } newp->fstype = strdup(mnttab_entry.mnt_fstype); if (newp->fstype == NULL) { /* * Out of memory */ fs_free_mount_list(newp); return (NULL); } newp->mntopts = strdup(mnttab_entry.mnt_mntopts); if (newp->mntopts == NULL) { /* * Out of memory */ fs_free_mount_list(newp); return (NULL); } newp->time = strdup(mnttab_entry.mnt_time); if (newp->time == NULL) { /* * Out of memory */ fs_free_mount_list(newp); return (NULL); } newp->major = mnttab_entry.mnt_major; newp->minor = mnttab_entry.mnt_minor; newp->next = NULL; return (newp); } /* create_extmntlist_entry */ static struct mnttab * create_mnttab_filter(char *resource, char *mountp, char *fstype, char *mntopts, char *time) { struct mnttab *search_entry; search_entry = (struct mnttab *)calloc((size_t)1, (size_t)sizeof (struct mnttab)); if (search_entry == NULL) { /* * Out of memory */ return (NULL); } if (resource != NULL) { search_entry->mnt_special = strdup(resource); if (search_entry->mnt_special == NULL) { /* * Out of memory */ free_mnttab_entry(search_entry); return (NULL); } } if (mountp != NULL) { search_entry->mnt_mountp = strdup(mountp); if (search_entry->mnt_mountp == NULL) { /* * Out of memory */ free_mnttab_entry(search_entry); return (NULL); } } if (fstype != NULL) { search_entry->mnt_fstype = strdup(fstype); if (search_entry->mnt_fstype == NULL) { /* * Out of memory */ free_mnttab_entry(search_entry); return (NULL); } } if (mntopts != NULL) { search_entry->mnt_mntopts = strdup(mntopts); if (search_entry->mnt_mntopts == NULL) { /* * Out of memory */ free_mnttab_entry(search_entry); return (NULL); } } if (time != NULL) { search_entry->mnt_time = strdup(time); if (search_entry->mnt_time == NULL) { /* * Out of memory */ free_mnttab_entry(search_entry); return (NULL); } } return (search_entry); } /* create_mnttab_filter */ /* * We will go through the /etc/mnttab entries to determine the * instances of overlayed file systems. We do this with the following * assumptions: * * 1.) Entries in mnttab are ordered in the way that the most recent * mounts are placed at the bottom of /etc/mnttab. Contract to be * filed: * 2.) Mnttab entries that are returned from all mnttab library * functions such as getmntent, getextmntent, and getmntany in the order * as they are found in /etc/mnttab. Goes along with assumption #1. * 3.) All automounted NFS file systems will have an autofs entry and * a NFS entry in /etc/mnttab with the same mount point. Autofs * entries can be ignored. * 4.) The device id (dev=) uniquely identifies a mounted file system * on a host. * * Algorithm explanation: * ---------------------- * For each mnt_list entry * 1.) Compare it to each /etc/mnttab entry starting at the point in mnttab * where the mnt_list entry mount is and look for matching mount points, * but ignore all "autofs" entries * If a two entries are found with the same mount point mark the mnt_list * entry as being overlayed. */ static void find_overlayed_filesystems(fs_mntlist_t *mnt_list, boolean_t filtered_list, int *errp) { boolean_t exit = B_FALSE; fs_mntlist_t *mnt_list_to_compare; fs_mntlist_t *tmp; *errp = 0; if (filtered_list == B_TRUE) { /* * Get the complete mount list */ mnt_list_to_compare = fs_get_mount_list(B_FALSE, errp); if (mnt_list_to_compare == NULL) { /* * If complete_mnt_list is NULL there are two * possibilites: * 1.) There are simply no entries in /etc/mnttab. * 2.) An error was encountered. errp will reflect * the error. */ return; } } else { mnt_list_to_compare = mnt_list; } tmp = mnt_list_to_compare; while (mnt_list != NULL) { if (!(strcmp(mnt_list->fstype, "autofs") == 0)) { char *dev_id; dev_id = fs_parse_optlist_for_option(mnt_list->mntopts, "dev=", errp); if (dev_id == NULL) { return; } exit = B_FALSE; while (tmp != NULL && exit == B_FALSE) { if (!(strcmp(tmp->fstype, "autofs")) == 0) { char *tmp_dev_id; tmp_dev_id = fs_parse_optlist_for_option( tmp->mntopts, "dev=", errp); if (tmp_dev_id == NULL) { return; } if (strcmp(tmp_dev_id, dev_id) == 0) { /* * Start searching for an * overlay here. */ mnt_list->overlayed = is_overlayed(tmp, mnt_list->mountp); exit = B_TRUE; } free(tmp_dev_id); } tmp = tmp->next; } /* while (tmp != NULL && exit == B_FALSE) */ free(dev_id); } /* if (!(strcmp(mnt_list->fstype, "autofs") == 0)) */ mnt_list = mnt_list->next; } /* while (mnt_list != NULL) */ if (filtered_list == B_TRUE) fs_free_mount_list(mnt_list_to_compare); } /* find_overlayed_filesystems */ static void free_mnttab_entry(struct mnttab *mnttab_entry) { free(mnttab_entry->mnt_special); free(mnttab_entry->mnt_mountp); free(mnttab_entry->mnt_fstype); free(mnttab_entry->mnt_mntopts); free(mnttab_entry->mnt_time); free(mnttab_entry); } /* free_mnttab_entry */ char * is_option(char *opt_string, char *opt, int *errp) { char *equalsign = "="; char *found_equalsign; char *return_val; *errp = 0; found_equalsign = strstr(opt, equalsign); /* * If found_equalsign is NULL then we did not find an equal sign * in the option we are to be looking for. */ if (found_equalsign == NULL) { if (strcmp(opt_string, opt) == 0) { /* * We have found the option so return with success. */ return_val = strdup(opt); if (return_val == NULL) { *errp = errno; return (NULL); } } else { return_val = NULL; } } else { int counter = 0; char *opt_found; char *value; opt_found = strstr(opt_string, opt); if (opt_found == NULL) { return_val = NULL; } else { size_t opt_string_len; size_t opt_len; size_t value_len; opt_string_len = strlen(opt_string); opt_len = strlen(opt); value_len = opt_string_len - opt_len; value = (char *)calloc((size_t)(value_len+1), (size_t)sizeof (char)); if (value == NULL) { /* * Out of memory */ *errp = ENOMEM; return (NULL); } while (counter <= (value_len-1)) { value[counter] = opt_string[opt_len+counter]; counter = counter + 1; } /* * Add the null terminating character. */ value[counter] = '\0'; return_val = value; } } /* else */ return (return_val); } /* is_option */ boolean_t is_overlayed(fs_mntlist_t *mnt_list, char *mountp) { boolean_t ret_val = B_FALSE; /* * The first entry in the complete_mnt_list is the same mounted * file system as the one we are trying to determine whether it is * overlayed or not. There is no need to compare these mounts. */ mnt_list = mnt_list->next; while (mnt_list != NULL && ret_val == B_FALSE) { if (!(strcmp(mnt_list->fstype, "autofs") == 0)) { if (strcmp(mnt_list->mountp, mountp) == 0) { ret_val = B_TRUE; } else { ret_val = B_FALSE; } } mnt_list = mnt_list->next; } return (ret_val); } /* is_overlayed */