#include <sys/types.h>
#include <sys/systm.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <sys/modctl.h>
#include <sys/sunddi.h>
#include <ipp/ipp.h>
#include <ipp/ipp_config.h>
#include <inet/common.h>
#include <ipp/dscpmk/dscpmk_impl.h>
#define D_SM_COMMENT "IPP dscpmk marker module"
uint8_t default_dscp_map[DSCPMK_ARRAY_COUNT] = {
0, 1, 2, 3,
4, 5, 6, 7,
8, 9, 10, 11,
12, 13, 14, 15,
16, 17, 18, 19,
20, 21, 22, 23,
24, 25, 26, 27,
28, 29, 30, 31,
32, 33, 34, 35,
36, 37, 38, 39,
40, 41, 42, 43,
44, 45, 46, 47,
48, 49, 50, 51,
52, 53, 54, 55,
56, 57, 58, 59,
60, 61, 62, 63
};
static int dscpmk_create_action(ipp_action_id_t, nvlist_t **, ipp_flags_t);
static int dscpmk_modify_action(ipp_action_id_t, nvlist_t **, ipp_flags_t);
static int dscpmk_destroy_action(ipp_action_id_t, ipp_flags_t);
static int dscpmk_info(ipp_action_id_t, int (*)(nvlist_t *, void *), void *,
ipp_flags_t);
static int dscpmk_invoke_action(ipp_action_id_t, ipp_packet_t *);
static int dscpmk_summ_statinit(ipp_action_id_t, dscpmk_data_t *);
static int dscpmk_update_stats(ipp_stat_t *, void *, int);
static int dscpmk_det_statinit(ipp_action_id_t, dscpmk_data_t *, int);
static int dscpmk_update_det_stats(ipp_stat_t *, void *, int);
ipp_ops_t dscpmk_ops = {
IPPO_REV,
dscpmk_create_action,
dscpmk_modify_action,
dscpmk_destroy_action,
dscpmk_info,
dscpmk_invoke_action
};
extern struct mod_ops mod_ippops;
static struct modlipp modlipp = {
&mod_ippops,
D_SM_COMMENT,
&dscpmk_ops
};
static struct modlinkage modlinkage = {
MODREV_1,
(void *)&modlipp,
NULL
};
int
_init(void)
{
return (mod_install(&modlinkage));
}
int
_fini(void)
{
return (mod_remove(&modlinkage));
}
int
_info(struct modinfo *modinfop)
{
return (mod_info(&modlinkage, modinfop));
}
static int
dscpmk_create_action(ipp_action_id_t aid, nvlist_t **nvlpp, ipp_flags_t flags)
{
nvlist_t *nvlp;
dscpmk_data_t *dscpmk_data;
char *next_action;
int err, cnt;
int32_t *tbl;
uint_t nelem = DSCPMK_ARRAY_COUNT;
uint32_t bstats;
ASSERT((nvlpp != NULL) && (*nvlpp != NULL));
nvlp = *nvlpp;
*nvlpp = NULL;
if ((dscpmk_data = kmem_zalloc(DSCPMK_DATA_SZ, KM_NOSLEEP)) == NULL) {
nvlist_free(nvlp);
return (ENOMEM);
}
if ((err = nvlist_lookup_string(nvlp, DSCPMK_NEXT_ACTION_NAME,
&next_action)) != 0) {
nvlist_free(nvlp);
dscpmk0dbg(("dscpmk_create_action: invalid config, " \
"next_action name missing\n"));
kmem_free(dscpmk_data, DSCPMK_DATA_SZ);
return (err);
}
if ((dscpmk_data->next_action = ipp_action_lookup(next_action))
== IPP_ACTION_INVAL) {
nvlist_free(nvlp);
dscpmk0dbg(("dscpmk_create_action: next_action "\
"invalid\n"));
kmem_free(dscpmk_data, DSCPMK_DATA_SZ);
return (EINVAL);
}
bcopy(default_dscp_map, dscpmk_data->dscp_map,
sizeof (default_dscp_map));
if ((err = nvlist_lookup_int32_array(nvlp, DSCPMK_DSCP_MAP,
&tbl, &nelem)) == 0) {
for (cnt = 0; cnt < DSCPMK_ARRAY_COUNT; cnt++) {
if ((tbl[cnt] != DSCPMK_UNCHANGED_DSCP) && (tbl[cnt] !=
dscpmk_data->dscp_map[cnt])) {
dscpmk_data->dscp_map[cnt] = tbl[cnt];
}
}
}
if ((err = nvlist_lookup_uint32(nvlp, IPP_ACTION_STATS_ENABLE, &bstats))
!= 0) {
dscpmk_data->summary_stats = B_FALSE;
} else {
dscpmk_data->summary_stats = (bstats != 0) ? B_TRUE : B_FALSE;
if (dscpmk_data->summary_stats) {
if ((err = dscpmk_summ_statinit(aid, dscpmk_data))
!= 0) {
nvlist_free(nvlp);
kmem_free(dscpmk_data, DSCPMK_DATA_SZ);
return (err);
}
}
}
for (cnt = 0; cnt < DSCPMK_ARRAY_COUNT; cnt++) {
dscpmk_data->dscp_stats[cnt].present = B_FALSE;
dscpmk_data->dscp_stats[cnt].npackets = 0;
}
if ((err = nvlist_lookup_uint32(nvlp, DSCPMK_DETAILED_STATS, &bstats))
!= 0) {
dscpmk_data->detailed_stats = B_FALSE;
} else {
dscpmk_data->detailed_stats = (bstats != 0) ? B_TRUE : B_FALSE;
if (dscpmk_data->detailed_stats) {
for (cnt = 0; cnt < DSCPMK_ARRAY_COUNT; cnt++) {
int val = dscpmk_data->dscp_map[cnt];
if (dscpmk_data->dscp_stats[val].present) {
continue;
}
dscpmk_data->dscp_stats[val].present = B_TRUE;
if ((err = dscpmk_det_statinit(aid, dscpmk_data,
val)) != 0) {
nvlist_free(nvlp);
kmem_free(dscpmk_data, DSCPMK_DATA_SZ);
return (err);
}
}
}
}
nvlist_free(nvlp);
if ((err = ipp_action_ref(aid, dscpmk_data->next_action, flags)) != 0) {
dscpmk0dbg(("dscpmk_create_action: ipp_action_ref " \
"returned with error %d\n", err));
if (dscpmk_data->summary_stats) {
ipp_stat_destroy(dscpmk_data->stats);
}
if (dscpmk_data->detailed_stats) {
for (cnt = 0; cnt < DSCPMK_ARRAY_COUNT; cnt++) {
if (dscpmk_data->dscp_stats[cnt].present) {
ipp_stat_destroy(
dscpmk_data->dscp_stats[cnt].stats);
}
}
}
kmem_free(dscpmk_data, DSCPMK_DATA_SZ);
return (err);
}
ipp_action_set_ptr(aid, (void *)dscpmk_data);
return (0);
}
static int
dscpmk_modify_action(ipp_action_id_t aid, nvlist_t **nvlpp, ipp_flags_t flags)
{
nvlist_t *nvlp;
int err = 0, cnt;
uint8_t config_type;
char *next_action_name;
uint32_t bstats;
uint_t nelem = DSCPMK_ARRAY_COUNT;
int32_t *tbl;
ipp_action_id_t next_action;
dscpmk_data_t *dscpmk_data;
ASSERT((nvlpp != NULL) && (*nvlpp != NULL));
nvlp = *nvlpp;
*nvlpp = NULL;
if ((err = nvlist_lookup_byte(nvlp, IPP_CONFIG_TYPE, &config_type))
!= 0) {
nvlist_free(nvlp);
dscpmk0dbg(("dscpmk_modify_action: invalid cfg. type\n"));
return (err);
}
if (config_type != IPP_SET) {
nvlist_free(nvlp);
dscpmk0dbg(("dscpmk_modify_action: invalid cfg. type " \
"%d\n", config_type));
return (EINVAL);
}
dscpmk_data = (dscpmk_data_t *)ipp_action_get_ptr(aid);
ASSERT(dscpmk_data != NULL);
if ((err = nvlist_lookup_string(nvlp, DSCPMK_NEXT_ACTION_NAME,
&next_action_name)) == 0) {
if ((next_action = ipp_action_lookup(next_action_name))
== IPP_ACTION_INVAL) {
nvlist_free(nvlp);
dscpmk0dbg(("dscpmk_modify_action: next_action "\
"invalid\n"));
return (EINVAL);
}
if ((err = ipp_action_ref(aid, next_action, flags)) != 0) {
nvlist_free(nvlp);
dscpmk0dbg(("dscpmk_modify_action: ipp_action_ref " \
"returned with error %d\n", err));
return (err);
}
err = ipp_action_unref(aid, dscpmk_data->next_action, flags);
ASSERT(err == 0);
dscpmk_data->next_action = next_action;
}
if ((err = nvlist_lookup_int32_array(nvlp, DSCPMK_DSCP_MAP,
&tbl, &nelem)) == 0) {
for (cnt = 0; cnt < DSCPMK_ARRAY_COUNT; cnt++) {
if ((tbl[cnt] != DSCPMK_UNCHANGED_DSCP) && (tbl[cnt] !=
dscpmk_data->dscp_map[cnt])) {
dscpmk_data->dscp_map[cnt] = tbl[cnt];
}
}
} else {
bcopy(default_dscp_map, dscpmk_data->dscp_map,
sizeof (default_dscp_map));
}
if ((err = nvlist_lookup_uint32(nvlp, IPP_ACTION_STATS_ENABLE, &bstats))
== 0) {
boolean_t val = (bstats != 0) ? B_TRUE : B_FALSE;
if (!dscpmk_data->summary_stats && val) {
if ((err = dscpmk_summ_statinit(aid, dscpmk_data))
!= 0) {
nvlist_free(nvlp);
return (err);
}
} else if (!val && dscpmk_data->summary_stats) {
ipp_stat_destroy(dscpmk_data->stats);
}
dscpmk_data->summary_stats = val;
}
if ((err = nvlist_lookup_uint32(nvlp, DSCPMK_DETAILED_STATS, &bstats))
== 0) {
boolean_t val = (bstats != 0) ? B_TRUE : B_FALSE;
if (dscpmk_data->detailed_stats && !val) {
for (cnt = 0; cnt < DSCPMK_ARRAY_COUNT; cnt++) {
if (dscpmk_data->dscp_stats[cnt].present) {
dscpmk_data->dscp_stats[cnt].present =
B_FALSE;
ipp_stat_destroy(dscpmk_data->
dscp_stats[cnt].stats);
}
}
}
dscpmk_data->detailed_stats = val;
}
if (dscpmk_data->detailed_stats) {
for (cnt = 0; cnt < DSCPMK_ARRAY_COUNT; cnt++) {
int val = dscpmk_data->dscp_map[cnt];
if (!dscpmk_data->dscp_stats[val].present) {
dscpmk_data->dscp_stats[val].present = B_TRUE;
if ((err = dscpmk_det_statinit(aid, dscpmk_data,
val)) != 0) {
nvlist_free(nvlp);
return (err);
}
}
}
}
nvlist_free(nvlp);
return (0);
}
static int
dscpmk_destroy_action(ipp_action_id_t aid, ipp_flags_t flags)
{
dscpmk_data_t *dscpmk_data;
int err, cnt;
dscpmk_data = (dscpmk_data_t *)ipp_action_get_ptr(aid);
ASSERT(dscpmk_data != NULL);
if (dscpmk_data->summary_stats) {
ipp_stat_destroy(dscpmk_data->stats);
}
if (dscpmk_data->detailed_stats) {
for (cnt = 0; cnt < DSCPMK_ARRAY_COUNT; cnt++) {
if (dscpmk_data->dscp_stats[cnt].present) {
ipp_stat_destroy(dscpmk_data->dscp_stats[cnt].
stats);
}
}
}
err = ipp_action_unref(aid, dscpmk_data->next_action, flags);
ASSERT(err == 0);
kmem_free(dscpmk_data, DSCPMK_DATA_SZ);
return (0);
}
static int
dscpmk_invoke_action(ipp_action_id_t aid, ipp_packet_t *packet)
{
dscpmk_data_t *dscpmk_data;
mblk_t *mp = NULL;
ip_priv_t *priv;
int err;
ASSERT(packet != NULL);
mp = ipp_packet_get_data(packet);
priv = (ip_priv_t *)ipp_packet_get_private(packet);
dscpmk_data = (dscpmk_data_t *)ipp_action_get_ptr(aid);
ASSERT(dscpmk_data != NULL);
if ((err = dscpmk_process(&mp, dscpmk_data, priv->proc)) != 0) {
return (err);
} else {
return (ipp_packet_next(packet, dscpmk_data->next_action));
}
}
static int
dscpmk_det_statinit(ipp_action_id_t aid, dscpmk_data_t *dscpmk_data, int val)
{
int err = 0;
dscpmk_dscp_stats_t *statp;
char stats_string[15];
(void) sprintf(stats_string, "dscpmk_dscp0x%x", val);
if ((err = ipp_stat_create(aid, stats_string, DSCPMK_DSCP_STATS_COUNT,
dscpmk_update_det_stats, dscpmk_data,
&dscpmk_data->dscp_stats[val].stats)) != 0) {
dscpmk0dbg(("dscpmk_det_statinit: ipp_stat_create returned "\
"with error %d\n", err));
return (err);
}
statp = (dscpmk_dscp_stats_t *)
(dscpmk_data->dscp_stats[val].stats)->ipps_data;
ASSERT(statp != NULL);
if ((err = ipp_stat_named_init(dscpmk_data->dscp_stats[val].stats,
"dscp", IPP_STAT_UINT32, &statp->dscp)) != 0) {
dscpmk0dbg(("dscpmk_det_statinit: ipp_stat_named_init "\
"returned with error %d\n", err));
return (err);
}
if ((err = ipp_stat_named_init(dscpmk_data->dscp_stats[val].stats,
"npackets", IPP_STAT_UINT64, &statp->npackets)) != 0) {
dscpmk0dbg(("dscpmk_det_statinit: ipp_stat_named_init "\
"returned with error %d\n", err));
return (err);
}
ipp_stat_install(dscpmk_data->dscp_stats[val].stats);
return (0);
}
static int
dscpmk_summ_statinit(ipp_action_id_t aid, dscpmk_data_t *dscpmk_data)
{
int err = 0;
dscpmk_stat_t *statp;
if ((err = ipp_stat_create(aid, DSCPMK_STATS_STRING, DSCPMK_STATS_COUNT,
dscpmk_update_stats, dscpmk_data, &dscpmk_data->stats)) != 0) {
dscpmk0dbg(("dscpmk_create_action: ipp_stat_create returned " \
"with error %d\n", err));
return (err);
}
statp = (dscpmk_stat_t *)(dscpmk_data->stats)->ipps_data;
ASSERT(statp != NULL);
if ((err = ipp_stat_named_init(dscpmk_data->stats, "npackets",
IPP_STAT_UINT64, &statp->npackets)) != 0) {
dscpmk0dbg(("dscpmk_summ_statinit: ipp_stat_named_init " \
"returned with error %d\n", err));
return (err);
}
if ((err = ipp_stat_named_init(dscpmk_data->stats, "dscp_changed",
IPP_STAT_UINT64, &statp->dscp_changed)) != 0) {
dscpmk0dbg(("dscpmk_summ_statinit: ipp_stat_named_init " \
"returned with error %d\n", err));
return (err);
}
if ((err = ipp_stat_named_init(dscpmk_data->stats, "dscp_unchanged",
IPP_STAT_UINT64, &statp->dscp_unchanged)) != 0) {
dscpmk0dbg(("dscpmk_summ_statinit: ipp_stat_named_init " \
"returned with error %d\n", err));
return (err);
}
if ((err = ipp_stat_named_init(dscpmk_data->stats, "ipackets",
IPP_STAT_UINT64, &statp->ipackets)) != 0) {
dscpmk0dbg(("dscpmk_summ_statinit: ipp_stat_named_init " \
"returned with error %d\n", err));
return (err);
}
if ((err = ipp_stat_named_init(dscpmk_data->stats, "epackets",
IPP_STAT_UINT64, &statp->epackets)) != 0) {
dscpmk0dbg(("dscpmk_summ_statinit: ipp_stat_named_init " \
"returned with error %d\n", err));
return (err);
}
ipp_stat_install(dscpmk_data->stats);
return (0);
}
static int
dscpmk_update_det_stats(ipp_stat_t *sp, void *arg, int rw)
{
dscpmk_data_t *dscpmk_data = (dscpmk_data_t *)arg;
dscpmk_dscp_stats_t *statp;
uint32_t count;
for (count = 0; count < DSCPMK_ARRAY_COUNT; count++) {
if (!dscpmk_data->dscp_stats[count].present)
continue;
statp = (dscpmk_dscp_stats_t *)
(dscpmk_data->dscp_stats[count].stats)->ipps_data;
ASSERT(statp != NULL);
(void) ipp_stat_named_op(&statp->npackets,
&dscpmk_data->dscp_stats[count].npackets, rw);
(void) ipp_stat_named_op(&statp->dscp, &count, rw);
}
return (0);
}
static int
dscpmk_update_stats(ipp_stat_t *sp, void *arg, int rw)
{
dscpmk_data_t *dscpmk_data = (dscpmk_data_t *)arg;
dscpmk_stat_t *snames = (dscpmk_stat_t *)sp->ipps_data;
ASSERT(dscpmk_data != NULL);
ASSERT(snames != NULL);
(void) ipp_stat_named_op(&snames->npackets, &dscpmk_data->npackets, rw);
(void) ipp_stat_named_op(&snames->dscp_changed, &dscpmk_data->changed,
rw);
(void) ipp_stat_named_op(&snames->dscp_unchanged,
&dscpmk_data->unchanged, rw);
(void) ipp_stat_named_op(&snames->ipackets, &dscpmk_data->ipackets, rw);
(void) ipp_stat_named_op(&snames->epackets, &dscpmk_data->epackets, rw);
return (0);
}
static int
dscpmk_info(ipp_action_id_t aid, int (*fn)(nvlist_t *, void *), void *arg,
ipp_flags_t flags)
{
nvlist_t *nvlp;
dscpmk_data_t *dscpmk_data;
char *next_action;
int err, cnt;
int32_t dscp_map[DSCPMK_ARRAY_COUNT];
ASSERT(fn != NULL);
dscpmk_data = (dscpmk_data_t *)ipp_action_get_ptr(aid);
ASSERT(dscpmk_data != NULL);
if ((err = nvlist_alloc(&nvlp, NV_UNIQUE_NAME, KM_NOSLEEP)) != 0) {
dscpmk0dbg(("dscpmk_info: error allocating memory\n"));
return (err);
}
if ((err = ipp_action_name(dscpmk_data->next_action,
&next_action)) != 0) {
dscpmk0dbg(("dscpmk_info: next action not available\n"));
nvlist_free(nvlp);
return (err);
}
if ((err = nvlist_add_string(nvlp, DSCPMK_NEXT_ACTION_NAME,
next_action)) != 0) {
dscpmk0dbg(("dscpmk_info: error adding next action\n"));
nvlist_free(nvlp);
kmem_free(next_action, (strlen(next_action) + 1));
return (err);
}
kmem_free(next_action, (strlen(next_action) + 1));
if ((err = nvlist_add_byte(nvlp, IPP_CONFIG_TYPE, IPP_SET)) != 0) {
dscpmk0dbg(("dscpmk_info: error adding config type\n"));
nvlist_free(nvlp);
return (err);
}
bcopy(dscpmk_data->dscp_map, dscp_map, sizeof (dscp_map));
for (cnt = 0; cnt < DSCPMK_ARRAY_COUNT; cnt++) {
dscp_map[cnt] = dscpmk_data->dscp_map[cnt];
}
if ((err = nvlist_add_int32_array(nvlp, DSCPMK_DSCP_MAP,
dscp_map, DSCPMK_ARRAY_COUNT)) != 0) {
dscpmk0dbg(("dscpmk_info: error adding dscp map\n"));
nvlist_free(nvlp);
return (err);
}
if ((err = nvlist_add_uint32(nvlp, IPP_ACTION_STATS_ENABLE,
(dscpmk_data->summary_stats ? 1 : 0))) != 0) {
dscpmk0dbg(("dscpmk_info: error adding stats status\n"));
nvlist_free(nvlp);
return (err);
}
if ((err = nvlist_add_uint32(nvlp, DSCPMK_DETAILED_STATS,
(dscpmk_data->detailed_stats ? 1 : 0))) != 0) {
dscpmk0dbg(("dscpmk_info: error adding det stats status\n"));
nvlist_free(nvlp);
return (err);
}
err = fn(nvlp, arg);
nvlist_free(nvlp);
return (err);
}