root/usr/src/cmd/tsol/lslabels/lslabels.c
/*
 * 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 2009 Sun Microsystems, Inc.  All rights reserved.
 * Use is subject to license terms.
 */


/*
 *      lslabels - Display all labels dominating the specified label.
 */

#include <errno.h>
#include <libintl.h>
#include <locale.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <stropts.h>

#include <sys/param.h>

#include <tsol/label.h>
#include <sys/tsol/label_macro.h>
#include <iso/limits_iso.h>

#if !defined(TEXT_DOMAIN)
#define TEXT_DOMAIN      "SYS_TEST"
#endif  /* !defined(TEXT_DOMAIN) */

int hflg = 0;                   /* true if hex output */

/*
 * Compartment mask macros.
 */

typedef uint32_t comp_chunk_t;

#define __NBWRD         (CHAR_BIT * sizeof (comp_chunk_t))
#define COMP_BITS       (CHAR_BIT * sizeof (Compartments_t))
#define compmask(n)     (1 << ((__NBWRD - 1) - ((n) % __NBWRD)))
#define compword(n)     ((n)/__NBWRD)

#define COMP_ADDSET(a, p)       ((comp_chunk_t *)(a))[compword(p)] |= \
                                    compmask(p)
#define COMP_DELSET(a, p)       ((comp_chunk_t *)(a))[compword(p)] &= \
                                    ~compmask(p)
#define COMP_ISMEMBER(a, p)     ((((comp_chunk_t *)(a))[compword(p)] & \
                                    compmask(p)) != 0)

/* Need functions to test if bit is on */


void
bitfinder(m_label_t label, int next_bit) {
        char *labelstr = NULL;

        Compartments_t *comps = &label.compartments;

        while (next_bit < COMP_BITS) {
                if (COMP_ISMEMBER(comps, next_bit)) {
                        bitfinder(label, next_bit + 1);
                        COMP_DELSET(comps, next_bit);

                        if (label_to_str(&label, &labelstr, M_LABEL,
                            LONG_NAMES) == 0) {
                                m_label_t *label2 = NULL;
                                int err;

                                if (str_to_label(labelstr, &label2, MAC_LABEL,
                                    L_NO_CORRECTION, &err) == 0) {
                                        if (!hflg) {
                                                (void) printf("%s\n", labelstr);
                                        } else {
                                                free(labelstr);
                                                (void) label_to_str(&label,
                                                    &labelstr, M_INTERNAL, 0);
                                                (void) printf("%s\n", labelstr);
                                        }
                                        m_label_free(label2);
                                }
                                free(labelstr);
                        }
                        bitfinder(label, next_bit + 1);
                        break;
                }
                next_bit++;
                }
}

static void
label_error(const char *ascii, const int err)
{
        if (errno == EINVAL) {
                switch (err) {
                case M_BAD_STRING:
                        (void) fprintf(stderr,
                            gettext("lslabels: bad string %s\n"), ascii);
                break;
                case M_BAD_LABEL:
                        (void) fprintf(stderr,
                            gettext("lslabels: bad previous label\n"));
                break;
                default:
                        (void) fprintf(stderr,
                            gettext("lslabels: parsing error found in "
                            "\"%s\" at position %d\n"), ascii, err);
                break;
                }
        } else {
                perror("lslabels");
        }
        exit(1);
        /*NOTREACHED*/
}

int
main(int argc, char **argv)
{
        int errflg = 0;                 /* true if arg error */
        m_label_t *label = NULL;        /* binary labels */
        char ascii[PIPE_BUF];           /* human readable label */
        char *labelstr = NULL;          /* external label to start from */
        int err = 0;                    /* label error */
        int c;
        int mode = M_LABEL;
        _Classification *level;

        (void) setlocale(LC_ALL, "");
        (void) textdomain(TEXT_DOMAIN);

        opterr = 0;
        while ((c = getopt(argc, argv, "h")) != EOF) {

                switch (c) {
                case 'h':
                        hflg++;
                        mode = M_INTERNAL;
                        break;

                default:
                        errflg++;
                        break;
                }
        }

        argc -= optind - 1;
        if (errflg || argc > 2) {

                (void) fprintf(stderr,
                    gettext("usage: %s [-h] [label]\n"),
                    argv[0]);
                exit(1);
                /*NOTREACHED*/
        }

        if (argc == 2) {
                /* use label on command line */

                (void) strlcpy(ascii, argv[optind], sizeof (ascii));
        } else {
                /* read label from standard input */
                if ((c = read(STDIN_FILENO, ascii, sizeof (ascii))) <= 0) {
                        perror(gettext("reading ASCII coded label"));
                        exit(1);
                        /*NOTREACHED*/
                }

                /*
                 * replace '\n' or (end of buffer) with end of string.
                 */
                ascii[c-1] = '\0';

                /*
                 * flush any remaining input past the size of the buffer.
                 */
                (void) ioctl(STDIN_FILENO, I_FLUSH, FLUSHR);
        }

        if (str_to_label(ascii, &label, MAC_LABEL, L_NO_CORRECTION,
            &err) == -1) {
                label_error(ascii, err);
        }
        if (label_to_str(label, &labelstr, mode,
            DEF_NAMES) == 0) {
                (void) printf("%s\n", labelstr);
        }

        level =  &label->classification.class_u.class_chunk;
        while (*level > 0) {
                bitfinder(*label, 0);
                *level -= 1;
        }
        m_label_free(label);

        return (0);             /* really exit(0); */
}