#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 VDS_DEBUG
#define DPRINTF(x) printf x
#else
#define DPRINTF(x)
#endif
struct vds_softc {
struct device sc_dv;
bus_space_tag_t sc_bustag;
bus_dma_tag_t sc_dmatag;
};
int vds_match(struct device *, void *, void *);
void vds_attach(struct device *, struct device *, void *);
const struct cfattach vds_ca = {
sizeof(struct vds_softc), vds_match, vds_attach
};
struct cfdriver vds_cd = {
NULL, "vds", DV_DULL
};
void vds_get_channel_endpoint(int, struct cbus_attach_args *);
int
vds_match(struct device *parent, void *match, void *aux)
{
struct cbus_attach_args *ca = aux;
if (strcmp(ca->ca_name, "virtual-disk-server") == 0)
return (1);
return (0);
}
void
vds_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 = "vds-port";
nca.ca_node = ca->ca_node;
nca.ca_bustag = ca->ca_bustag;
nca.ca_dmatag = ca->ca_dmatag;
vds_get_channel_endpoint(arc, &nca);
config_found(self, &nca, cbus_print);
}
}
}
void
vds_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;
}
}
}