blackmagic-sys 0.1.0

Bindings to Blackmagic-debug host functions
Documentation
/*
 * This file is part of the Black Magic Debug project.
 *
 * Copyright (C) 2011  Black Sphere Technologies Ltd.
 * Written by Gareth McMullin <gareth@blacksphere.co.nz>
 * Copyright (C) 2018  Uwe Bonnes (non@elektron.ikp.physik.tu-darmstadt.de)
 * Copyright (C) 2022-2023 1BitSquared <info@1bitsquared.com>
 * Modified by Rachel Mant <git@dragonmux.network>
 *
 * 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 3 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, see <http://www.gnu.org/licenses/>.
 */

#ifndef PLATFORMS_HOSTED_FTDI_BMP_H
#define PLATFORMS_HOSTED_FTDI_BMP_H

#include <ftdi.h>

#include "cli.h"
#include "jtagtap.h"

#include "bmp_hosted.h"

#include "probe_info.h"

typedef struct data_desc {
	uint16_t data_low;
	uint16_t ddr_low;
	uint16_t data_high;
	uint16_t ddr_high;
} data_desc_s;

typedef struct pin_settings {
	uint8_t set_data_low;
	uint8_t clr_data_low;
	uint8_t set_data_high;
	uint8_t clr_data_high;
} pin_settings_s;

typedef struct cable_desc {
	int vendor;
	int product;
	int interface;
	/* Initial (C|D)(Bus|Ddr) values for additional pins.
	 * MPSSE_CS|DI|DO|SK are initialized accordig to mode.*/
	data_desc_s init;
	/* MPSSE command to read TMS/SWDIO in bitbanging SWD.
	 * In many cases this is the TMS port, so then use "GET_PIN_LOW".*/
	uint8_t bb_swdio_in_port_cmd;
	/* bus bit to read TMS/SWDIO in bitbanging SWD.
	 * In many cases this is the TMS port, so then use "MPSSE_CS".*/
	uint8_t bb_swdio_in_pin;
	/* Bus data to allow bitbanging switched SWD read.
	 * TMS is routed to bb_swdio_in_port/pin.*/
	pin_settings_s bb_swd_read;
	/* Bus data to allow bitbanging switched SWD write.
	 * TMS is routed to MPSSE_CS.*/
	pin_settings_s bb_swd_write;
	/* dbus_data, dbus_ddr, cbus_data, cbus_ddr value to assert nRST.
	 *	E.g. with CBUS Pin 1 low,
	 *	give data_high = ~PIN1, ddr_high = PIN1 */
	data_desc_s assert_nrst;
	/*  Bus_data, dbus_ddr, cbus_data, cbus_ddr value to release nRST.
	 *	E.g. with CBUS Pin 1 floating with internal pull up,
	 *	give data_high = PIN1, ddr_high = ~PIN1 */
	data_desc_s deassert_nrst;
	/* Command to read back NRST. If 0, port from assert_nrst is used*/
	uint8_t nrst_get_port_cmd;
	/* PIN to read back as NRST. if 0 port from assert_nrst is ised.
	*  Use PINX if active high, use Complement (~PINX) if active low*/
	uint8_t nrst_get_pin;
	/* Bbus data for pure MPSSE SWD read.
	 * Use together with swd_write if by some bits on DBUS,
	 * SWDIO can be routed to TDI and TDO.
	 * If both mpsse_swd_read|write and
	 * bitbang_swd_dbus_read_data/bitbang_tms_in_port_cmd/bitbang_tms_in_pin
	 * are provided, pure MPSSE SWD is chosen.
	 * If neither a complete set of swd_read|write or
	 * bitbang_swd_dbus_read_data/bitbang_tms_in_port_cmd/bitbang_tms_in_pin
	 * are provided, SWD can not be done.
	 * swd_read.set_data_low ==  swd_write.set_data_low == MPSSE_DO
	 * indicated resistor SWD and inhibits Jtag.*/
	pin_settings_s mpsse_swd_read;
	/* dbus data for pure MPSSE SWD write.*/
	pin_settings_s mpsse_swd_write;
	/* dbus data for jtag.*/
	pin_settings_s jtag;
	/* Command to read port to check target voltage.*/
	uint8_t target_voltage_cmd;
	/* Pin to check target voltage.*/
	uint8_t target_voltage_pin;
	/* USB readable description of the device.*/
	char *description;
	/* Command line argument to -c option to select this device.*/
	char *name;
} cable_desc_s;

typedef struct ftdi_mpsse_cmd {
	uint8_t command;
	uint8_t length[2];
} ftdi_mpsse_cmd_s;

typedef struct ftdi_mpsse_cmd_bits {
	uint8_t command;
	uint8_t length;
} ftdi_mpsse_cmd_bits_s;

extern const cable_desc_s cable_desc[];
extern cable_desc_s active_cable;
extern data_desc_s active_state;

#define ftdi_buffer_write_arr(array) ftdi_buffer_write(array, sizeof(array))
#define ftdi_buffer_write_val(value) ftdi_buffer_write(&(value), sizeof(value))
#define ftdi_buffer_read_arr(array)  ftdi_buffer_read(array, sizeof(array))
#define ftdi_buffer_read_val(value)  ftdi_buffer_read(&(value), sizeof(value))

bool ftdi_bmp_init(bmda_cli_options_s *cl_opts);
bool ftdi_lookup_adapter_from_vid_pid(bmda_cli_options_s *cl_opts, const probe_info_s *probe);
bool ftdi_lookup_adaptor_descriptor(bmda_cli_options_s *cl_opts, const probe_info_s *probe);
bool ftdi_swd_init(void);
bool ftdi_jtag_init(void);
void ftdi_buffer_flush(void);
size_t ftdi_buffer_write(const void *buffer, size_t size);
size_t ftdi_buffer_read(void *buffer, size_t size);
const char *ftdi_target_voltage(void);
void ftdi_jtag_tdi_tdo_seq(uint8_t *data_out, bool final_tms, const uint8_t *data_in, size_t clock_cycles);
bool ftdi_swd_possible(void);
void ftdi_max_frequency_set(uint32_t freq);
uint32_t libftdi_max_frequency_get(void);
void libftdi_nrst_set_val(bool assert);
bool ftdi_nrst_get_val(void);

#define MPSSE_SK 1
#define PIN0     1
#define MPSSE_DO 2
#define PIN1     2
#define MPSSE_DI 4
#define PIN2     4
#define MPSSE_CS 8
#define PIN3     8
#define PIN4     0x10
#define PIN5     0x20
#define PIN6     0x40
#define PIN7     0x80

#endif /* PLATFORMS_HOSTED_FTDI_BMP_H */