dld_flow.c revision 2b24ab6b3865caeede9eeb9db6b83e1d89dcd1ea
/*
* 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.
*/
/*
* Flows ioctls implementation.
*/
#include <sys/cred.h>
#include <sys/dld.h>
#include <sys/mac_provider.h>
#include <sys/mac_client.h>
#include <sys/mac_client_priv.h>
/*
* Implements flow add, remove, modify ioctls.
*/
int
dld_add_flow(datalink_id_t linkid, char *flow_name, flow_desc_t *flow_desc,
mac_resource_props_t *mrp)
{
return (mac_link_flow_add(linkid, flow_name, flow_desc, mrp));
}
int
dld_remove_flow(char *flow_name)
{
return (mac_link_flow_remove(flow_name));
}
int
dld_modify_flow(char *flow_name, mac_resource_props_t *mrp)
{
return (mac_link_flow_modify(flow_name, mrp));
}
/*
* Callback function and structure used by dld_walk_flow().
*/
typedef struct flowinfo_state_s {
int fi_bufsize;
int fi_nflows;
uchar_t *fi_fl;
} flowinfo_state_t;
static int
dld_walk_flow_cb(mac_flowinfo_t *finfo, void *arg)
{
flowinfo_state_t *statep = arg;
dld_flowinfo_t fi;
if (statep->fi_bufsize < sizeof (dld_flowinfo_t))
return (ENOSPC);
(void) strlcpy(fi.fi_flowname, finfo->fi_flow_name,
sizeof (fi.fi_flowname));
fi.fi_linkid = finfo->fi_link_id;
fi.fi_flow_desc = finfo->fi_flow_desc;
fi.fi_resource_props = finfo->fi_resource_props;
if (copyout(&fi, statep->fi_fl, sizeof (fi)) != 0) {
return (EFAULT);
}
statep->fi_nflows++;
statep->fi_bufsize -= sizeof (dld_flowinfo_t);
statep->fi_fl += sizeof (dld_flowinfo_t);
return (0);
}
/*
* Implements flow walk ioctl.
* Retrieves a specific flow or a list of flows from the specified link.
* ENOSPC is returned a bigger buffer is needed.
*/
int
dld_walk_flow(dld_ioc_walkflow_t *wf, intptr_t uaddr, cred_t *credp)
{
flowinfo_state_t state;
mac_flowinfo_t finfo;
int err = 0;
/* For now, one can only view flows from the global zone. */
if (crgetzoneid(credp) != GLOBAL_ZONEID)
return (EPERM);
state.fi_bufsize = wf->wf_len;
state.fi_fl = (uchar_t *)uaddr + sizeof (*wf);
state.fi_nflows = 0;
if (wf->wf_name[0] == '\0') {
err = mac_link_flow_walk(wf->wf_linkid, dld_walk_flow_cb,
&state);
} else {
err = mac_link_flow_info(wf->wf_name, &finfo);
if (err != 0)
return (err);
err = dld_walk_flow_cb(&finfo, &state);
}
wf->wf_nflows = state.fi_nflows;
return (err);
}