464 lines
9.7 KiB
C
464 lines
9.7 KiB
C
/* MSPDebug - debugging tool for MSP430 MCUs
|
|
* Copyright (C) 2009, 2010 Daniel Beer
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include <string.h>
|
|
|
|
#include "output.h"
|
|
#include "output_util.h"
|
|
#include "dis.h"
|
|
#include "simio.h"
|
|
#include "simio_cpu.h"
|
|
#include "simio_device.h"
|
|
|
|
#include "simio_tracer.h"
|
|
#include "simio_timer.h"
|
|
#include "simio_wdt.h"
|
|
#include "simio_hwmult.h"
|
|
#include "simio_gpio.h"
|
|
#include "simio_console.h"
|
|
|
|
static const struct simio_class *const class_db[] = {
|
|
&simio_tracer,
|
|
&simio_timer,
|
|
&simio_wdt,
|
|
&simio_hwmult,
|
|
&simio_gpio,
|
|
&simio_console
|
|
};
|
|
|
|
/* Simulator data. We keep a list of devices on the bus, and the special
|
|
* function registers.
|
|
*
|
|
* Currently, MCLK and SMCLK are tied together, and ACLK runs at a fixed
|
|
* ratio of 1:256 with MCLK. aclk_counter counts fractional cycles.
|
|
*/
|
|
static struct list_node device_list;
|
|
static uint8_t sfr_data[16];
|
|
static int aclk_counter;
|
|
|
|
static void destroy_device(struct simio_device *dev)
|
|
{
|
|
list_remove(&dev->node);
|
|
dev->type->destroy(dev);
|
|
}
|
|
|
|
void simio_init(void)
|
|
{
|
|
list_init(&device_list);
|
|
simio_reset();
|
|
}
|
|
|
|
void simio_exit(void)
|
|
{
|
|
while (!LIST_EMPTY(&device_list))
|
|
destroy_device((struct simio_device *)device_list.next);
|
|
}
|
|
|
|
static const struct simio_class *find_class(const char *name)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < ARRAY_LEN(class_db); i++) {
|
|
const struct simio_class *t = class_db[i];
|
|
|
|
if (!strcasecmp(t->name, name))
|
|
return t;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static struct simio_device *find_device(const char *name)
|
|
{
|
|
struct list_node *n;
|
|
|
|
for (n = device_list.next; n != &device_list; n = n->next) {
|
|
struct simio_device *dev = (struct simio_device *)n;
|
|
|
|
if (!strcasecmp(dev->name, name))
|
|
return dev;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static int cmd_add(char **arg_text)
|
|
{
|
|
const char *type_text = get_arg(arg_text);
|
|
const char *name_text = get_arg(arg_text);
|
|
const struct simio_class *type;
|
|
struct simio_device *dev;
|
|
|
|
if (!(name_text && type_text)) {
|
|
printc_err("simio add: device class and name must be "
|
|
"specified.\n");
|
|
return -1;
|
|
}
|
|
|
|
if (find_device(name_text)) {
|
|
printc_err("simio add: device name is not unique: %s\n",
|
|
name_text);
|
|
return -1;
|
|
}
|
|
|
|
type = find_class(type_text);
|
|
if (!type) {
|
|
printc_err("simio add: unknown type.\n");
|
|
return -1;
|
|
}
|
|
|
|
dev = type->create(arg_text);
|
|
if (!dev) {
|
|
printc_err("simio add: failed to create device.\n");
|
|
return -1;
|
|
}
|
|
|
|
list_insert(&dev->node, &device_list);
|
|
strncpy(dev->name, name_text, sizeof(dev->name));
|
|
dev->name[sizeof(dev->name) - 1] = 0;
|
|
|
|
printc_dbg("Added new device \"%s\" of type \"%s\".\n",
|
|
dev->name, dev->type->name);
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_del(char **arg_text)
|
|
{
|
|
const char *name_text = get_arg(arg_text);
|
|
struct simio_device *dev;
|
|
|
|
if (!name_text) {
|
|
printc_err("simio del: device name must be specified.\n");
|
|
return -1;
|
|
}
|
|
|
|
dev = find_device(name_text);
|
|
if (!dev) {
|
|
printc_err("simio del: no such device: %s\n", name_text);
|
|
return -1;
|
|
}
|
|
|
|
destroy_device(dev);
|
|
printc_dbg("Destroyed device \"%s\".\n", name_text);
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_devices(char **arg_text)
|
|
{
|
|
struct list_node *n;
|
|
|
|
(void)arg_text;
|
|
|
|
for (n = device_list.next; n != &device_list; n = n->next) {
|
|
struct simio_device *dev = (struct simio_device *)n;
|
|
int irq = -1;
|
|
|
|
if (dev->type->check_interrupt)
|
|
irq = dev->type->check_interrupt(dev);
|
|
|
|
printc(" %-10s (type %s", dev->name, dev->type->name);
|
|
if (irq < 0)
|
|
printc(")\n");
|
|
else
|
|
printc(", IRQ pending: %d)\n", irq);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_classes(char **arg_text)
|
|
{
|
|
struct vector v;
|
|
int i;
|
|
|
|
(void)arg_text;
|
|
|
|
vector_init(&v, sizeof(const char *));
|
|
for (i = 0; i < ARRAY_LEN(class_db); i++) {
|
|
if (vector_push(&v, &class_db[i]->name, 1) < 0) {
|
|
printc_err("simio classes: can't allocate memory\n");
|
|
vector_destroy(&v);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
printc("Available device classes:\n");
|
|
namelist_print(&v);
|
|
vector_destroy(&v);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_help(char **arg_text)
|
|
{
|
|
const char *name = get_arg(arg_text);
|
|
const struct simio_class *type;
|
|
|
|
if (!name) {
|
|
printc_err("simio help: you must specify a device class\n");
|
|
return -1;
|
|
}
|
|
|
|
type = find_class(name);
|
|
if (!type) {
|
|
printc_err("simio help: unknown device class: %s\n", name);
|
|
return -1;
|
|
}
|
|
|
|
printc("\x1b[1mDEVICE CLASS: %s\x1b[0m\n\n%s\n", type->name, type->help);
|
|
return 0;
|
|
}
|
|
|
|
static int cmd_config(char **arg_text)
|
|
{
|
|
const char *name = get_arg(arg_text);
|
|
const char *param = get_arg(arg_text);
|
|
struct simio_device *dev;
|
|
|
|
if (!(name && param)) {
|
|
printc_err("simio config: you must specify a device name and "
|
|
"a parameter\n");
|
|
return -1;
|
|
}
|
|
|
|
dev = find_device(name);
|
|
if (!dev) {
|
|
printc_err("simio config: no such device: %s\n", name);
|
|
return -1;
|
|
}
|
|
|
|
if (!dev->type->config) {
|
|
printc_err("simio config: no configuration parameters are "
|
|
"defined for this device\n");
|
|
return -1;
|
|
}
|
|
|
|
return dev->type->config(dev, param, arg_text);
|
|
}
|
|
|
|
static int cmd_info(char **arg_text)
|
|
{
|
|
const char *name = get_arg(arg_text);
|
|
struct simio_device *dev;
|
|
|
|
if (!name) {
|
|
printc_err("simio info: you must specify a device name\n");
|
|
return -1;
|
|
}
|
|
|
|
dev = find_device(name);
|
|
if (!dev) {
|
|
printc_err("simio info: no such device: %s\n", name);
|
|
return -1;
|
|
}
|
|
|
|
if (!dev->type->info) {
|
|
printc_err("simio config: no information available\n");
|
|
return -1;
|
|
}
|
|
|
|
return dev->type->info(dev);
|
|
}
|
|
|
|
int cmd_simio(char **arg_text)
|
|
{
|
|
const char *subcmd = get_arg(arg_text);
|
|
static const struct {
|
|
const char *name;
|
|
int (*func)(char **arg_text);
|
|
} cmd_table[] = {
|
|
{"add", cmd_add},
|
|
{"del", cmd_del},
|
|
{"devices", cmd_devices},
|
|
{"classes", cmd_classes},
|
|
{"help", cmd_help},
|
|
{"config", cmd_config},
|
|
{"info", cmd_info}
|
|
};
|
|
int i;
|
|
|
|
if (!subcmd) {
|
|
printc_err("simio: a subcommand is required\n");
|
|
return -1;
|
|
}
|
|
|
|
for (i = 0; i < ARRAY_LEN(cmd_table); i++)
|
|
if (!strcasecmp(cmd_table[i].name, subcmd))
|
|
return cmd_table[i].func(arg_text);
|
|
|
|
printc_err("simio: unknown subcommand: %s\n", subcmd);
|
|
return -1;
|
|
}
|
|
|
|
void simio_reset(void)
|
|
{
|
|
struct list_node *n;
|
|
|
|
memset(sfr_data, 0, sizeof(sfr_data));
|
|
aclk_counter = 0;
|
|
|
|
for (n = device_list.next; n != &device_list; n = n->next) {
|
|
struct simio_device *dev = (struct simio_device *)n;
|
|
const struct simio_class *type = dev->type;
|
|
|
|
if (type->reset)
|
|
type->reset(dev);
|
|
}
|
|
}
|
|
|
|
#define IO_REQUEST_FUNC(name, method, datatype) \
|
|
int name(address_t addr, datatype data) { \
|
|
struct list_node *n; \
|
|
int ret = 1; \
|
|
\
|
|
for (n = device_list.next; n != &device_list; n = n->next) { \
|
|
struct simio_device *dev = (struct simio_device *)n; \
|
|
const struct simio_class *type = dev->type; \
|
|
\
|
|
if (type->method) { \
|
|
int r = type->method(dev, addr, data); \
|
|
\
|
|
if (r < ret) \
|
|
ret = r; \
|
|
} \
|
|
} \
|
|
\
|
|
return ret; \
|
|
}
|
|
#define IO_REQUEST_FUNC_S(name, method, datatype) \
|
|
static IO_REQUEST_FUNC(name, method, datatype)
|
|
|
|
IO_REQUEST_FUNC(simio_write, write, uint16_t)
|
|
IO_REQUEST_FUNC(simio_read_device, read, uint16_t *)
|
|
IO_REQUEST_FUNC_S(simio_write_b_device, write_b, uint8_t)
|
|
IO_REQUEST_FUNC_S(simio_read_b_device, read_b, uint8_t *)
|
|
|
|
int simio_read(address_t addr, uint16_t *data)
|
|
{
|
|
addr &= ~1;
|
|
if (addr < 16) {
|
|
*data = ((uint16_t)sfr_data[addr]) |
|
|
(((uint16_t)sfr_data[addr + 1]) << 8);
|
|
return 0;
|
|
}
|
|
|
|
*data = 0;
|
|
return simio_read_device(addr, data);
|
|
}
|
|
|
|
int simio_write_b(address_t addr, uint8_t data)
|
|
{
|
|
if (addr < 16) {
|
|
sfr_data[addr] = data;
|
|
return 0;
|
|
}
|
|
|
|
return simio_write_b_device(addr, data);
|
|
}
|
|
|
|
int simio_read_b(address_t addr, uint8_t *data)
|
|
{
|
|
if (addr < 16) {
|
|
*data = sfr_data[addr];
|
|
return 0;
|
|
}
|
|
|
|
*data = 0;
|
|
return simio_read_b_device(addr, data);
|
|
}
|
|
|
|
int simio_check_interrupt(void)
|
|
{
|
|
int irq = -1;
|
|
struct list_node *n;
|
|
|
|
for (n = device_list.next; n != &device_list; n = n->next) {
|
|
struct simio_device *dev = (struct simio_device *)n;
|
|
const struct simio_class *type = dev->type;
|
|
|
|
if (type->check_interrupt) {
|
|
int i = type->check_interrupt(dev);
|
|
|
|
if (i > irq)
|
|
irq = i;
|
|
}
|
|
}
|
|
|
|
return irq;
|
|
}
|
|
|
|
void simio_ack_interrupt(int irq)
|
|
{
|
|
struct list_node *n;
|
|
|
|
for (n = device_list.next; n != &device_list; n = n->next) {
|
|
struct simio_device *dev = (struct simio_device *)n;
|
|
const struct simio_class *type = dev->type;
|
|
|
|
if (type->ack_interrupt)
|
|
type->ack_interrupt(dev, irq);
|
|
}
|
|
}
|
|
|
|
void simio_step(uint16_t status_register, int cycles)
|
|
{
|
|
int clocks[SIMIO_NUM_CLOCKS] = {0};
|
|
struct list_node *n;
|
|
|
|
aclk_counter += cycles;
|
|
|
|
clocks[SIMIO_MCLK] = cycles;
|
|
clocks[SIMIO_SMCLK] = cycles;
|
|
clocks[SIMIO_ACLK] = aclk_counter >> 8;
|
|
|
|
aclk_counter &= 0xff;
|
|
|
|
if (status_register & MSP430_SR_CPUOFF)
|
|
clocks[SIMIO_MCLK] = 0;
|
|
|
|
if (status_register & MSP430_SR_SCG1)
|
|
clocks[SIMIO_SMCLK] = 0;
|
|
|
|
if (status_register & MSP430_SR_OSCOFF)
|
|
clocks[SIMIO_ACLK] = 0;
|
|
|
|
for (n = device_list.next; n != &device_list; n = n->next) {
|
|
struct simio_device *dev = (struct simio_device *)n;
|
|
const struct simio_class *type = dev->type;
|
|
|
|
if (type->step)
|
|
type->step(dev, status_register, clocks);
|
|
}
|
|
}
|
|
|
|
uint8_t simio_sfr_get(address_t which)
|
|
{
|
|
if (which > sizeof(sfr_data))
|
|
return 0;
|
|
|
|
return sfr_data[which];
|
|
}
|
|
|
|
void simio_sfr_modify(address_t which, uint8_t mask, uint8_t bits)
|
|
{
|
|
if (which > sizeof(sfr_data))
|
|
return;
|
|
|
|
sfr_data[which] = (sfr_data[which] & ~mask) | bits;
|
|
}
|