#include <locale.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <strings.h>
#include <wchar.h>
#include <sys/mman.h>
#include <tsol/label.h>
#include "clnt.h"
#include "labeld.h"
#include <sys/tsol/label_macro.h>
#if !defined(TEXT_DOMAIN)
#define TEXT_DOMAIN "SYS_TEST"
#endif
static bslabel_t slow;
static bslabel_t shigh;
static bclear_t clrlow, clrhigh;
static char *sstring;
static size_t ssize;
static int
return_string(char **string, int str_len, char *val)
{
char *cpyptr;
size_t val_len = strlen(val) + 1;
if (*string == NULL) {
if ((*string = malloc(val_len)) == NULL)
return (0);
} else if (val_len > str_len) {
**string = '\0';
return (0);
}
cpyptr = *string;
bcopy(val, cpyptr, val_len);
return (val_len);
}
void
set_label_view(uint_t *callflags, uint_t flags)
{
if (flags&VIEW_INTERNAL) {
*callflags |= LABELS_VIEW_INTERNAL;
} else if (flags&VIEW_EXTERNAL) {
*callflags |= LABELS_VIEW_EXTERNAL;
}
}
int
alloc_string(char **string, size_t size, char val)
{
if (*string == NULL) {
if ((*string = malloc(ALLOC_CHUNK)) == NULL)
return (0);
} else {
if ((*string = realloc(*string, size + ALLOC_CHUNK)) == NULL) {
**string = val;
return (0);
}
}
**string = val;
return (ALLOC_CHUNK);
}
#define slcall callp->param.acall.cargs.bsltos_arg
#define slret callp->param.aret.rvals.bsltos_ret
ssize_t
bsltos(const bslabel_t *label, char **string, size_t str_len,
int flags)
{
labeld_data_t call;
labeld_data_t *callp = &call;
size_t bufsize = sizeof (labeld_data_t);
size_t datasize = CALL_SIZE(bsltos_call_t, 0);
int rval;
if (!BLTYPE(label, SUN_SL_ID)) {
return (-1);
}
call.callop = BSLTOS;
slcall.label = *label;
slcall.flags = (flags&NO_CLASSIFICATION) ? LABELS_NO_CLASS : 0;
slcall.flags |= (flags&SHORT_CLASSIFICATION ||
!(flags&LONG_CLASSIFICATION)) ? LABELS_SHORT_CLASS : 0;
slcall.flags |= (flags&SHORT_WORDS && !(flags&LONG_WORDS)) ?
LABELS_SHORT_WORDS : 0;
set_label_view(&slcall.flags, flags);
if ((rval = __call_labeld(&callp, &bufsize, &datasize)) == SUCCESS) {
if (callp->reterr != 0)
return (-1);
rval = return_string(string, str_len, slret.slabel);
if (callp != &call)
(void) munmap((void *)callp, bufsize);
return (rval);
} else if (rval == NOSERVER) {
if (!BLTYPE(&slow, SUN_SL_ID)) {
BSLLOW(&slow);
BSLHIGH(&shigh);
}
if (BLEQUAL(label, &slow)) {
return (return_string(string, str_len, ADMIN_LOW));
} else if (BLEQUAL(label, &shigh)) {
return (return_string(string, str_len, ADMIN_HIGH));
}
}
return (-1);
}
#undef slcall
#undef slret
#define clrcall callp->param.acall.cargs.bcleartos_arg
#define clrret callp->param.aret.rvals.bcleartos_ret
ssize_t
bcleartos(const bclear_t *clearance, char **string, size_t str_len,
int flags)
{
labeld_data_t call;
labeld_data_t *callp = &call;
size_t bufsize = sizeof (labeld_data_t);
size_t datasize = CALL_SIZE(bcleartos_call_t, 0);
int rval;
if (!BLTYPE(clearance, SUN_CLR_ID)) {
return (-1);
}
call.callop = BCLEARTOS;
clrcall.clear = *clearance;
clrcall.flags = (flags&SHORT_CLASSIFICATION ||
!(flags&LONG_CLASSIFICATION)) ? LABELS_SHORT_CLASS : 0;
clrcall.flags |= (flags&SHORT_WORDS && !(flags&LONG_WORDS)) ?
LABELS_SHORT_WORDS : 0;
set_label_view(&clrcall.flags, flags);
if ((rval = __call_labeld(&callp, &bufsize, &datasize)) == SUCCESS) {
if (callp->reterr != 0)
return (-1);
rval = return_string(string, str_len, clrret.cslabel);
if (callp != &call)
(void) munmap((void *)callp, bufsize);
return (rval);
} else if (rval == NOSERVER) {
if (!BLTYPE(&clrlow, SUN_CLR_ID)) {
BCLEARLOW(&clrlow);
BCLEARHIGH(&clrhigh);
}
if (BLEQUAL(clearance, &clrlow)) {
return (return_string(string, str_len, ADMIN_LOW));
} else if (BLEQUAL(clearance, &clrhigh)) {
return (return_string(string, str_len, ADMIN_HIGH));
}
}
return (-1);
}
#undef clrcall
#undef clrret
char *
sbsltos(const bslabel_t *label, size_t len)
{
ssize_t slen;
wchar_t *wstring;
int wccount;
if (ssize == 0) {
if ((ssize = alloc_string(&sstring, ssize, 's')) == 0)
return (NULL);
}
again:
if ((slen = bsltos(label, &sstring, ssize,
(SHORT_CLASSIFICATION | LONG_WORDS))) <= 0) {
if (slen == 0) {
if (*sstring == '\0') {
int newsize;
if ((newsize = alloc_string(&sstring, ssize,
's')) == 0) {
return (NULL);
}
ssize += newsize;
goto again;
}
}
return (NULL);
}
if (len == 0) {
return (sstring);
} else if (len < MIN_SL_LEN) {
return (NULL);
}
if ((wstring = malloc(slen * sizeof (wchar_t))) == NULL)
return (NULL);
if ((wccount = mbstowcs(wstring, sstring, slen - 1)) == -1) {
free(wstring);
return (NULL);
}
if (wccount > len) {
wchar_t *clipp = wstring + (len - 2);
clipp[0] = L'<';
clipp[1] = L'-';
clipp[2] = L'\0';
while (wcstombs(NULL, wstring, 0) >= ssize) {
int newsize;
if ((newsize = alloc_string(&sstring, ssize, 's')) ==
0) {
return (NULL);
}
ssize += newsize;
}
if ((wccount = wcstombs(sstring, wstring, ssize)) == -1) {
free(wstring);
return (NULL);
}
}
free(wstring);
return (sstring);
}
char *
sbcleartos(const bclear_t *clearance, size_t len)
{
ssize_t slen;
wchar_t *wstring;
int wccount;
if (ssize == 0) {
if ((ssize = alloc_string(&sstring, ssize, 'c')) == 0)
return (NULL);
}
again:
if ((slen = bcleartos(clearance, &sstring, ssize,
(SHORT_CLASSIFICATION | LONG_WORDS))) <= 0) {
if (slen == 0) {
if (*sstring == '\0') {
int newsize;
if ((newsize = alloc_string(&sstring, ssize,
'c')) == 0) {
return (NULL);
}
ssize += newsize;
goto again;
}
}
return (NULL);
}
if (len == 0) {
return (sstring);
} else if (len < MIN_CLR_LEN) {
return (NULL);
}
if ((wstring = malloc(slen * sizeof (wchar_t))) == NULL)
return (NULL);
if ((wccount = mbstowcs(wstring, sstring, slen - 1)) == -1) {
free(wstring);
return (NULL);
}
if (wccount > len) {
wchar_t *clipp = wstring + (len - 2);
clipp[0] = L'<';
clipp[1] = L'-';
clipp[2] = L'\0';
while (wcstombs(NULL, wstring, 0) >= ssize) {
int newsize;
if ((newsize = alloc_string(&sstring, ssize, 'c')) ==
0) {
free(wstring);
return (NULL);
}
ssize += newsize;
}
if ((wccount = wcstombs(sstring, wstring, ssize)) == -1) {
free(wstring);
return (NULL);
}
}
free(wstring);
return (sstring);
}