blob: b86390c7b6a0277d87968a77bb503b8a068dcf09 [file] [log] [blame]
/*
* Copyright (c) 2019 The Fuchsia Authors.
*
* Permission to use, copy, modify, and/or distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
#include "sim.h"
#include <zircon/status.h>
#include <memory>
#include "bus.h"
#include "chip.h"
#include "common.h"
#include "debug.h"
#include "src/connectivity/wlan/drivers/testing/lib/sim-device/device.h"
#define BUS_OP(bus) bus->bus_priv.sim->sim_fw
static const struct brcmf_bus_ops brcmf_sim_bus_ops = {
.get_bus_type = []() { return BRCMF_BUS_TYPE_SIM; },
.preinit = [](brcmf_bus* bus) { return BUS_OP(bus)->BusPreinit(); },
.stop = [](brcmf_bus* bus) { return BUS_OP(bus)->BusStop(); },
.txdata = [](brcmf_bus* bus, brcmf_netbuf* netbuf) { return BUS_OP(bus)->BusTxData(netbuf); },
.txctl = [](brcmf_bus* bus, unsigned char* msg,
uint len) { return BUS_OP(bus)->BusTxCtl(msg, len); },
.rxctl = [](brcmf_bus* bus, unsigned char* msg, uint len,
int* rxlen_out) { return BUS_OP(bus)->BusRxCtl(msg, len, rxlen_out); },
.gettxq = [](brcmf_bus* bus) { return BUS_OP(bus)->BusGetTxQueue(); },
.wowl_config = [](brcmf_bus* bus, bool enabled) { return BUS_OP(bus)->BusWowlConfig(enabled); },
.get_ramsize = [](brcmf_bus* bus) { return BUS_OP(bus)->BusGetRamsize(); },
.get_memdump = [](brcmf_bus* bus, void* data,
size_t len) { return BUS_OP(bus)->BusGetMemdump(data, len); },
.get_fwname =
[](brcmf_bus* bus, uint chip, uint chiprev, unsigned char* fw_name) {
return BUS_OP(bus)->BusGetFwName(chip, chiprev, fw_name);
},
.get_bootloader_macaddr =
[](brcmf_bus* bus, uint8_t* mac_addr) {
return BUS_OP(bus)->BusGetBootloaderMacAddr(mac_addr);
},
.device_add = wlan_sim_device_add};
#undef BUS_OP
// Get device-specific information
zx_status_t brcmf_sim_probe(struct brcmf_bus* bus) {
uint32_t chip, chiprev;
bus->bus_priv.sim->sim_fw->GetChipInfo(&chip, &chiprev);
bus->chip = chip;
bus->chiprev = chiprev;
bus->bus_priv.sim->settings = brcmf_get_module_param(BRCMF_BUS_TYPE_SIM, chip, chiprev);
if (bus->bus_priv.sim->settings == nullptr) {
BRCMF_ERR("Failed to get device parameters\n");
return ZX_ERR_INTERNAL;
}
return ZX_OK;
}
// Allocate necessary memory and initialize simulator-specific structures
zx_status_t brcmf_sim_register(brcmf_pub* drvr, std::unique_ptr<brcmf_bus>* out_bus) {
zx_status_t status = ZX_OK;
auto simdev = std::make_unique<brcmf_simdev>();
auto bus_if = std::make_unique<brcmf_bus>();
// Initialize inter-structure pointers
simdev->sim_fw = std::make_unique<SimFirmware>();
bus_if->bus_priv.sim = simdev.get();
BRCMF_DBG(SIM, "Registering simulator target\n");
status = brcmf_sim_probe(bus_if.get());
if (status != ZX_OK) {
BRCMF_ERR("sim_probe failed: %s\n", zx_status_get_string(status));
return status;
}
bus_if->ops = &brcmf_sim_bus_ops;
status = brcmf_attach(drvr, bus_if.get(), simdev->settings);
if (status != ZX_OK) {
BRCMF_ERR("brcmf_attach failed\n");
return status;
}
// Here is where we would likely simulate loading the firmware into the target. For now,
// we don't try.
status = brcmf_bus_started(drvr);
if (status != ZX_OK) {
BRCMF_ERR("Failed to start (simulated) bus\n");
return status;
}
simdev.release();
*out_bus = std::move(bus_if);
return ZX_OK;
}
void brcmf_sim_exit(brcmf_bus* bus) {
delete bus->bus_priv.sim;
bus->bus_priv.sim = nullptr;
}