#include <sys/param.h>
#include <sys/buf.h>
#include <sys/device.h>
#include <sys/malloc.h>
#include <sys/systm.h>
#include <machine/autoconf.h>
#include <machine/hypervisor.h>
#include <machine/mdesc.h>
#include <sparc64/dev/cbusvar.h>
#ifdef VSW_DEBUG
#define DPRINTF(x) printf x
#else
#define DPRINTF(x)
#endif
struct vsw_softc {
struct device sc_dv;
bus_space_tag_t sc_bustag;
bus_dma_tag_t sc_dmatag;
};
int vsw_match(struct device *, void *, void *);
void vsw_attach(struct device *, struct device *, void *);
const struct cfattach vsw_ca = {
sizeof(struct vsw_softc), vsw_match, vsw_attach
};
struct cfdriver vsw_cd = {
NULL, "vsw", DV_DULL
};
void vsw_get_channel_endpoint(int, struct cbus_attach_args *);
int
vsw_match(struct device *parent, void *match, void *aux)
{
struct cbus_attach_args *ca = aux;
if (strcmp(ca->ca_name, "virtual-network-switch") == 0)
return (1);
return (0);
}
void
vsw_attach(struct device *parent, struct device *self, void *aux)
{
struct cbus_attach_args *ca = aux;
struct cbus_attach_args nca;
struct md_header *hdr;
struct md_element *elem;
const char *name_blk;
const char *str;
int idx;
int arc;
printf("\n");
hdr = (struct md_header *)mdesc;
elem = (struct md_element *)(mdesc + sizeof(struct md_header));
name_blk = mdesc + sizeof(struct md_header) + hdr->node_blk_sz;
idx = ca->ca_idx;
for (; elem[idx].tag != 'E'; idx++) {
str = name_blk + elem[idx].name_offset;
if (elem[idx].tag != 'a' || strcmp(str, "fwd") != 0)
continue;
arc = elem[idx].d.val;
str = name_blk + elem[arc].name_offset;
if (strcmp(str, "virtual-device-port") == 0) {
bzero(&nca, sizeof(nca));
nca.ca_name = "network";
nca.ca_node = ca->ca_node;
nca.ca_bustag = ca->ca_bustag;
nca.ca_dmatag = ca->ca_dmatag;
vsw_get_channel_endpoint(arc, &nca);
config_found(self, &nca, cbus_print);
}
}
}
void
vsw_get_channel_endpoint(int idx, struct cbus_attach_args *ca)
{
struct md_header *hdr;
struct md_element *elem;
const char *name_blk;
const char *str;
int arc;
hdr = (struct md_header *)mdesc;
elem = (struct md_element *)(mdesc + sizeof(struct md_header));
name_blk = mdesc + sizeof(struct md_header) + hdr->node_blk_sz;
ca->ca_idx = idx;
ca->ca_id = -1;
ca->ca_tx_ino = -1;
ca->ca_rx_ino = -1;
for (; elem[idx].tag != 'E'; idx++) {
str = name_blk + elem[idx].name_offset;
if (elem[idx].tag != 'a' || strcmp(str, "fwd") != 0)
continue;
arc = elem[idx].d.val;
str = name_blk + elem[arc].name_offset;
if (strcmp(str, "channel-endpoint") == 0) {
ca->ca_id = mdesc_get_prop_val(arc, "id");
ca->ca_tx_ino = mdesc_get_prop_val(arc, "tx-ino");
ca->ca_rx_ino = mdesc_get_prop_val(arc, "rx-ino");
return;
}
}
}