/***************************************************************************
 *   Copyright (C) 2009 by Mathias Kuester                                 *
 *   mkdorg@users.sourceforge.net                                          *
 *                                                                         *
 *   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.,                                       *
 *   59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.             *
 ***************************************************************************/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif

#include <helper/jim.h>

#include "target.h"
#include "target_type.h"
#include "register.h"
#include "dsp563xx.h"
#include "dsp563xx_once.h"

#define DSP563XX_JTAG_INS_LEN		4

#define JTAG_STATUS_NORMAL		0x01
#define JTAG_STATUS_STOPWAIT		0x05
#define JTAG_STATUS_BUSY		0x09
#define JTAG_STATUS_DEBUG		0x0d

#define JTAG_INSTR_EXTEST		0x00
#define JTAG_INSTR_SAMPLE_PRELOAD	0x01
#define JTAG_INSTR_IDCODE		0x02
#define JTAG_INSTR_CLAMP		0x03
#define JTAG_INSTR_HIZ			0x04
#define JTAG_INSTR_ENABLE_ONCE		0x06
#define JTAG_INSTR_DEBUG_REQUEST	0x07
#define JTAG_INSTR_BYPASS		0x0F

/* forward declarations */
static int dsp563xx_write_ir_u8(struct jtag_tap *tap, uint8_t * ir_in, uint8_t ir_out,
		int ir_len, int rti);

/* IR and DR functions */
static int dsp563xx_jtag_sendinstr(struct jtag_tap *tap, uint8_t * ir_in, uint8_t ir_out);

#define ASM_REG_R_R0	0x607000
#define ASM_REG_R_R1	0x617000
#define ASM_REG_R_R2	0x627000
#define ASM_REG_R_R3	0x637000
#define ASM_REG_R_R4	0x647000
#define ASM_REG_R_R5	0x657000
#define ASM_REG_R_R6	0x667000
#define ASM_REG_R_R7	0x677000

#define ASM_REG_W_R0	0x60F400
#define ASM_REG_W_R1	0x61F400
#define ASM_REG_W_R2	0x62F400
#define ASM_REG_W_R3	0x63F400
#define ASM_REG_W_R4	0x64F400
#define ASM_REG_W_R5	0x65F400
#define ASM_REG_W_R6	0x66F400
#define ASM_REG_W_R7	0x67F400

#define ASM_REG_R_N0	0x707000
#define ASM_REG_R_N1	0x717000
#define ASM_REG_R_N2	0x727000
#define ASM_REG_R_N3	0x737000
#define ASM_REG_R_N4	0x747000
#define ASM_REG_R_N5	0x757000
#define ASM_REG_R_N6	0x767000
#define ASM_REG_R_N7	0x777000

#define ASM_REG_W_N0	0x70F400
#define ASM_REG_W_N1	0x71F400
#define ASM_REG_W_N2	0x72F400
#define ASM_REG_W_N3	0x73F400
#define ASM_REG_W_N4	0x74F400
#define ASM_REG_W_N5	0x75F400
#define ASM_REG_W_N6	0x76F400
#define ASM_REG_W_N7	0x77F400

#define ASM_REG_R_M0	0x057020	/* control register m[0..7] */
#define ASM_REG_R_M1	0x057021
#define ASM_REG_R_M2	0x057022
#define ASM_REG_R_M3	0x057023
#define ASM_REG_R_M4	0x057024
#define ASM_REG_R_M5	0x057025
#define ASM_REG_R_M6	0x057026
#define ASM_REG_R_M7	0x057027

#define ASM_REG_W_M0	0x05F420
#define ASM_REG_W_M1	0x05F421
#define ASM_REG_W_M2	0x05F422
#define ASM_REG_W_M3	0x05F423
#define ASM_REG_W_M4	0x05F424
#define ASM_REG_W_M5	0x05F425
#define ASM_REG_W_M6	0x05F426
#define ASM_REG_W_M7	0x05F427

#define ASM_REG_R_X0	0x447000
#define ASM_REG_R_X1	0x457000

#define ASM_REG_W_X0	0x44F400
#define ASM_REG_W_X1	0x45F400

#define ASM_REG_R_Y0	0x467000
#define ASM_REG_R_Y1	0x477000

#define ASM_REG_W_Y0	0x46F400
#define ASM_REG_W_Y1	0x47F400

#define ASM_REG_R_A0	0x507000
#define ASM_REG_R_A1	0x547000
#define ASM_REG_R_A2	0x527000

#define ASM_REG_W_A0	0x50F400
#define ASM_REG_W_A1	0x54F400
#define ASM_REG_W_A2	0x52F400

#define ASM_REG_R_B0	0x517000
#define ASM_REG_R_B1	0x557000
#define ASM_REG_R_B2	0x537000

#define ASM_REG_W_B0	0x51F400
#define ASM_REG_W_B1	0x55F400
#define ASM_REG_W_B2	0x53F400

#define ASM_REG_R_VBA	0x057030	/* control register */
#define ASM_REG_W_VBA	0x05F430

#define ASM_REG_R_OMR	0x05703A	/* control register */
#define ASM_REG_W_OMR	0x05F43A

#define ASM_REG_R_EP	0x05702A
#define ASM_REG_W_EP	0x05F42A

#define ASM_REG_R_SC	0x057031	/* stack counter */
#define ASM_REG_W_SC	0x05F431

#define ASM_REG_R_SZ	0x057038	/* stack size */
#define ASM_REG_W_SZ	0x05F438

#define ASM_REG_R_SR	0x057039	/* control register, status register */
#define ASM_REG_W_SR	0x05F439

#define ASM_REG_R_SP	0x05703B	/* control register, stack pointer */
#define ASM_REG_W_SP	0x05F43B

#define ASM_REG_R_SSH	0x05703C	/* control register, system stack high */
#define ASM_REG_W_SSH	0x05743C

#define ASM_REG_R_SSL	0x05703D	/* control register, system stack low */
#define ASM_REG_W_SSL	0x05F43D

#define ASM_REG_R_LA	0x05703E	/* control register, loop address */
#define ASM_REG_W_LA	0x05F43E

#define ASM_REG_R_LC	0x05703F	/* control register, loop count */
#define ASM_REG_W_LC	0x05F43F

#define ASM_REG_R_PC	0x000000
#define ASM_REG_W_PC	0x000000

static const struct
{
	unsigned id;
	char *name;
	unsigned bits;
	uint32_t r_cmd;
	uint32_t w_cmd;
} dsp563xx_regs[] =
{
	/* *INDENT-OFF* */
	{0, "r0", 24, ASM_REG_R_R0, ASM_REG_W_R0},
	{1, "r1", 24, ASM_REG_R_R1, ASM_REG_W_R1},
	{2, "r2", 24, ASM_REG_R_R2, ASM_REG_W_R2},
	{3, "r3", 24, ASM_REG_R_R3, ASM_REG_W_R3},
	{4, "r4", 24, ASM_REG_R_R4, ASM_REG_W_R4},
	{5, "r5", 24, ASM_REG_R_R5, ASM_REG_W_R5},
	{6, "r6", 24, ASM_REG_R_R6, ASM_REG_W_R6},
	{7, "r7", 24, ASM_REG_R_R7, ASM_REG_W_R7},
	{8, "n0", 24, ASM_REG_R_N0, ASM_REG_W_N0},
	{9, "n1", 24, ASM_REG_R_N1, ASM_REG_W_N1},
	{10, "n2", 24, ASM_REG_R_N2, ASM_REG_W_N2},
	{11, "n3", 24, ASM_REG_R_N3, ASM_REG_W_N3},
	{12, "n4", 24, ASM_REG_R_N4, ASM_REG_W_N4},
	{13, "n5", 24, ASM_REG_R_N5, ASM_REG_W_N5},
	{14, "n6", 24, ASM_REG_R_N6, ASM_REG_W_N6},
	{15, "n7", 24, ASM_REG_R_N7, ASM_REG_W_N7},
	{16, "m0", 24, ASM_REG_R_M0, ASM_REG_W_M0},
	{17, "m1", 24, ASM_REG_R_M1, ASM_REG_W_M1},
	{18, "m2", 24, ASM_REG_R_M2, ASM_REG_W_M2},
	{19, "m3", 24, ASM_REG_R_M3, ASM_REG_W_M3},
	{20, "m4", 24, ASM_REG_R_M4, ASM_REG_W_M4},
	{21, "m5", 24, ASM_REG_R_M5, ASM_REG_W_M5},
	{22, "m6", 24, ASM_REG_R_M6, ASM_REG_W_M6},
	{23, "m7", 24, ASM_REG_R_M7, ASM_REG_W_M7},
	{24, "x0", 24, ASM_REG_R_X0, ASM_REG_W_X0},
	{25, "x1", 24, ASM_REG_R_X1, ASM_REG_W_X1},
	{26, "y0", 24, ASM_REG_R_Y0, ASM_REG_W_Y0},
	{27, "y1", 24, ASM_REG_R_Y1, ASM_REG_W_Y1},
	{28, "a0", 24, ASM_REG_R_A0, ASM_REG_W_A0},
	{29, "a1", 24, ASM_REG_R_A1, ASM_REG_W_A1},
	{30, "a2", 8, ASM_REG_R_A2, ASM_REG_W_A2},
	{31, "b0", 24, ASM_REG_R_B0, ASM_REG_W_B0},
	{32, "b1", 24, ASM_REG_R_B1, ASM_REG_W_B1},
	{33, "b2", 8, ASM_REG_R_B2, ASM_REG_W_B2},
	{34, "omr", 24, ASM_REG_R_OMR, ASM_REG_W_OMR},
	{35, "vba", 24, ASM_REG_R_VBA, ASM_REG_W_VBA},
	{36, "ep", 24, ASM_REG_R_EP, ASM_REG_W_EP},
	{37, "sc", 24, ASM_REG_R_SC, ASM_REG_W_SC},
	{38, "sz", 24, ASM_REG_R_SZ, ASM_REG_W_SZ},
	{39, "sr", 24, ASM_REG_R_SR, ASM_REG_W_SR},
	{40, "sp", 24, ASM_REG_R_SP, ASM_REG_W_SP},
	{41, "la", 24, ASM_REG_R_LA, ASM_REG_W_LA},
	{42, "lc", 24, ASM_REG_R_LC, ASM_REG_W_LC},
	{43, "pc", 24, ASM_REG_R_PC, ASM_REG_W_PC}
	/* *INDENT-ON* */
};

static int dsp563xx_get_gdb_reg_list(struct target *target, struct reg **reg_list[],
			      int *reg_list_size)
{
	struct dsp563xx_common *dsp563xx = target_to_dsp563xx(target);
	int i;

	if (target->state != TARGET_HALTED)
	{
		return ERROR_TARGET_NOT_HALTED;
	}

	*reg_list_size = DSP563XX_NUMCOREREGS;
	*reg_list = malloc(sizeof(struct reg *) * (*reg_list_size));

	for (i = 0; i < DSP563XX_NUMCOREREGS; i++)
	{
		(*reg_list)[i] = &dsp563xx->core_cache->reg_list[i];
	}

	return ERROR_OK;

}

static int dsp563xx_read_core_reg(struct target *target, int num)
{
	uint32_t reg_value;
	struct dsp563xx_core_reg *dsp563xx_core_reg;
	struct dsp563xx_common *dsp563xx = target_to_dsp563xx(target);

	if ((num < 0) || (num >= DSP563XX_NUMCOREREGS))
		return ERROR_INVALID_ARGUMENTS;

	dsp563xx_core_reg = dsp563xx->core_cache->reg_list[num].arch_info;
	reg_value = dsp563xx->core_regs[num];
	buf_set_u32(dsp563xx->core_cache->reg_list[num].value, 0, 32, reg_value);
	dsp563xx->core_cache->reg_list[num].valid = 1;
	dsp563xx->core_cache->reg_list[num].dirty = 0;

	return ERROR_OK;
}

static int dsp563xx_write_core_reg(struct target *target, int num)
{
	uint32_t reg_value;
	struct dsp563xx_core_reg *dsp563xx_core_reg;
	struct dsp563xx_common *dsp563xx = target_to_dsp563xx(target);

	if ((num < 0) || (num >= DSP563XX_NUMCOREREGS))
		return ERROR_INVALID_ARGUMENTS;

	reg_value = buf_get_u32(dsp563xx->core_cache->reg_list[num].value, 0, 32);
	dsp563xx_core_reg = dsp563xx->core_cache->reg_list[num].arch_info;
	dsp563xx->core_regs[num] = reg_value;
	dsp563xx->core_cache->reg_list[num].valid = 1;
	dsp563xx->core_cache->reg_list[num].dirty = 0;

	return ERROR_OK;
}

static int dsp563xx_target_create(struct target *target, Jim_Interp * interp)
{
	struct dsp563xx_common *dsp563xx = calloc(1, sizeof(struct dsp563xx_common));

	dsp563xx->jtag_info.tap = target->tap;
	target->arch_info = dsp563xx;
	dsp563xx->read_core_reg = dsp563xx_read_core_reg;
	dsp563xx->write_core_reg = dsp563xx_write_core_reg;

	return ERROR_OK;
}

static int dsp563xx_get_core_reg(struct reg *reg)
{
	int retval = 0;

	LOG_DEBUG("%s", __FUNCTION__);

	struct dsp563xx_core_reg *dsp563xx_reg = reg->arch_info;
	struct target *target = dsp563xx_reg->target;
	struct dsp563xx_common *dsp563xx = target_to_dsp563xx(target);

	if (target->state != TARGET_HALTED)
	{
		return ERROR_TARGET_NOT_HALTED;
	}

	retval = dsp563xx->read_core_reg(target, dsp563xx_reg->num);

	return retval;
}

static int dsp563xx_set_core_reg(struct reg *reg, uint8_t * buf)
{
	LOG_DEBUG("%s", __FUNCTION__);

	struct dsp563xx_core_reg *dsp563xx_reg = reg->arch_info;
	struct target *target = dsp563xx_reg->target;
	uint32_t value = buf_get_u32(buf, 0, 32);

	if (target->state != TARGET_HALTED)
	{
		return ERROR_TARGET_NOT_HALTED;
	}

	buf_set_u32(reg->value, 0, reg->size, value);
	reg->dirty = 1;
	reg->valid = 1;

	return ERROR_OK;
}

static int dsp563xx_save_context(struct target *target)
{
	int i;
	uint32_t data = 0;
	struct dsp563xx_common *dsp563xx = target_to_dsp563xx(target);
	struct dsp563xx_core_reg *arch_info;

	for (i = 0; i < DSP563XX_NUMCOREREGS - 1; i++)
	{

//              if (!dsp563xx->core_cache->reg_list[i].valid)
		{
			arch_info = dsp563xx->core_cache->reg_list[i].arch_info;
			dsp563xx_once_execute_dw_ir(target->tap, arch_info->r_cmd,
						    0xfffffc);
			dsp563xx_once_execute_sw_ir(target->tap, 0x000000);
			dsp563xx_once_reg_read(target->tap, DSP563XX_ONCE_OGDBR,
					       &data);
			dsp563xx->core_regs[i] = data;
			dsp563xx->read_core_reg(target, i);
		}
	}

	/* read pc */
	dsp563xx_once_reg_read(target->tap, DSP563XX_ONCE_OPABEX, &data);
	dsp563xx->core_regs[i] = data;
	dsp563xx->read_core_reg(target, i);

	return ERROR_OK;
}

static int dsp563xx_restore_context(struct target *target)
{
	int i;
	struct dsp563xx_common *dsp563xx = target_to_dsp563xx(target);
	struct dsp563xx_core_reg *arch_info;

	for (i = 0; i < DSP563XX_NUMCOREREGS - 1; i++)
	{
		if (dsp563xx->core_cache->reg_list[i].dirty)
		{
			arch_info = dsp563xx->core_cache->reg_list[i].arch_info;

			dsp563xx->write_core_reg(target, i);

			dsp563xx_once_execute_dw_ir(target->tap, arch_info->w_cmd,
						    dsp563xx->core_regs[i]);
			dsp563xx_once_execute_sw_ir(target->tap, 0x000000);
		}
	}

	return ERROR_OK;
}

static const struct reg_arch_type dsp563xx_reg_type = {
	.get = dsp563xx_get_core_reg,
	.set = dsp563xx_set_core_reg,
};

static int dsp563xx_init_target(struct command_context *cmd_ctx, struct target *target)
{
	/* get pointers to arch-specific information */
	struct dsp563xx_common *dsp563xx = target_to_dsp563xx(target);

	struct reg_cache **cache_p = register_get_last_cache_p(&target->reg_cache);
	struct reg_cache *cache = malloc(sizeof(struct reg_cache));
	struct reg *reg_list = malloc(sizeof(struct reg) * DSP563XX_NUMCOREREGS);
	struct dsp563xx_core_reg *arch_info =
		malloc(sizeof(struct dsp563xx_core_reg) * DSP563XX_NUMCOREREGS);
	int i;

	LOG_DEBUG("%s", __FUNCTION__);

	/* Build the process context cache */
	cache->name = "dsp563xx registers";
	cache->next = NULL;
	cache->reg_list = reg_list;
	cache->num_regs = DSP563XX_NUMCOREREGS;
	(*cache_p) = cache;
	dsp563xx->core_cache = cache;

	for (i = 0; i < DSP563XX_NUMCOREREGS; i++)
	{
		arch_info[i].num = dsp563xx_regs[i].id;
		arch_info[i].name = dsp563xx_regs[i].name;
		arch_info[i].size = dsp563xx_regs[i].bits;
		arch_info[i].r_cmd = dsp563xx_regs[i].r_cmd;
		arch_info[i].w_cmd = dsp563xx_regs[i].w_cmd;
		arch_info[i].target = target;
		arch_info[i].dsp563xx_common = dsp563xx;
		reg_list[i].name = dsp563xx_regs[i].name;
		reg_list[i].size = dsp563xx_regs[i].bits;
		reg_list[i].value = calloc(1, 4);
		reg_list[i].dirty = 0;
		reg_list[i].valid = 0;
		reg_list[i].type = &dsp563xx_reg_type;
		reg_list[i].arch_info = &arch_info[i];
	}

	return ERROR_OK;
}

static int dsp563xx_arch_state(struct target *target)
{
	LOG_DEBUG("%s", __FUNCTION__);
	return ERROR_OK;
}

static int dsp563xx_jtag_status(struct target *target, uint8_t * status)
{
	uint8_t ir_in;

	ir_in = 0;

	dsp563xx_jtag_sendinstr(target->tap, &ir_in, JTAG_INSTR_ENABLE_ONCE);
	dsp563xx_execute_queue();

	*status = ir_in;

	return ERROR_OK;
}

static int dsp563xx_jtag_debug_request(struct target *target)
{
	uint8_t ir_in = 0;
	uint32_t retry = 0;

	while (ir_in != JTAG_STATUS_DEBUG)
	{
		dsp563xx_jtag_sendinstr(target->tap, &ir_in,
					JTAG_INSTR_DEBUG_REQUEST);
		dsp563xx_execute_queue();
		LOG_DEBUG("JTAG CMD 7 res: %02X", ir_in);
		dsp563xx_jtag_sendinstr(target->tap, &ir_in, JTAG_INSTR_ENABLE_ONCE);
		dsp563xx_execute_queue();
		LOG_DEBUG("JTAG CMD 6 res: %02X", ir_in);

		if (retry++ == 100)
			return ERROR_TARGET_FAILURE;
	}

	if (ir_in != JTAG_STATUS_DEBUG)
	{
		return ERROR_TARGET_FAILURE;
	}

	return ERROR_OK;
}

static int dsp563xx_poll(struct target *target)
{
	uint8_t jtag_status;
	uint32_t once_status;

	dsp563xx_jtag_status(target, &jtag_status);

	if ((jtag_status & 1) != 1)
	{
		target->state = TARGET_UNKNOWN;
		LOG_ERROR
			("jtag status contains invalid mode value - communication failure");
		return ERROR_TARGET_FAILURE;
	}

	if (jtag_status != JTAG_STATUS_DEBUG)
	{
		target->state = TARGET_RUNNING;
	}

	dsp563xx_once_reg_read(target->tap, DSP563XX_ONCE_OSCR, &once_status);

	if ((once_status & DSP563XX_ONCE_OSCR_DEBUG_M) == DSP563XX_ONCE_OSCR_DEBUG_M)
	{
		target->state = TARGET_HALTED;

	}

	return ERROR_OK;
}

static int dsp563xx_halt(struct target *target)
{
	uint8_t jtag_status;
	uint32_t once_status;
	struct dsp563xx_common *dsp563xx = target_to_dsp563xx(target);

	if (target->state == TARGET_HALTED)
	{
		LOG_DEBUG("target was already halted");
		return ERROR_OK;
	}

	if (target->state == TARGET_UNKNOWN)
	{
		LOG_WARNING("target was in unknown state when halt was requested");
	}

//      if ( jtag_status != 0x0d )
	{
		dsp563xx_jtag_debug_request(target);

		/* store pipeline register */
		dsp563xx_once_reg_read(target->tap, DSP563XX_ONCE_OPILR,
				       &dsp563xx->pipeline_context.once_opilr);
		dsp563xx_once_reg_read(target->tap, DSP563XX_ONCE_OPDBR,
				       &dsp563xx->pipeline_context.once_opdbr);

		dsp563xx_save_context(target);

		dsp563xx_jtag_status(target, &jtag_status);
		LOG_DEBUG("%02X", jtag_status);
		dsp563xx_once_reg_read(target->tap, DSP563XX_ONCE_OSCR,
				       &once_status);
		LOG_DEBUG("%02X", (unsigned) once_status);
	}

	LOG_DEBUG("target->state: %s", target_state_name(target));

	LOG_DEBUG("%s", __FUNCTION__);

	return ERROR_OK;
}

#define DSP563XX_ASM_CMD_JUMP	0x0AF080

static int dsp563xx_resume(struct target *target, int current, uint32_t address,
		    int handle_breakpoints, int debug_execution)
{
	struct dsp563xx_common *dsp563xx = target_to_dsp563xx(target);

	LOG_DEBUG("%s", __FUNCTION__);

	dsp563xx_restore_context(target);

	if (current)
	{
		/* restore pipeline registers and go */
		dsp563xx_once_reg_write(target->tap, DSP563XX_ONCE_OPILR,
					dsp563xx->pipeline_context.once_opilr);
		dsp563xx_once_reg_write(target->tap,
					DSP563XX_ONCE_OPDBR | DSP563XX_ONCE_OCR_EX |
					DSP563XX_ONCE_OCR_GO,
					dsp563xx->pipeline_context.once_opdbr);
	}
	else
	{
		/* set to go register and jump */
		dsp563xx_once_reg_write(target->tap, DSP563XX_ONCE_OPDBR,
					DSP563XX_ASM_CMD_JUMP);
		dsp563xx_once_reg_write(target->tap,
					DSP563XX_ONCE_PDBGOTO | DSP563XX_ONCE_OCR_EX
					| DSP563XX_ONCE_OCR_GO, address);
	}

	target->state = TARGET_RUNNING;

	return ERROR_OK;
}

static int dsp563xx_step(struct target *target, int current, uint32_t address,
		  int handle_breakpoints)
{
	uint32_t once_status;
	uint32_t dr_in, cnt;
	struct dsp563xx_common *dsp563xx = target_to_dsp563xx(target);

	if (target->state != TARGET_HALTED)
	{
		LOG_DEBUG("target was not halted");
		return ERROR_OK;
	}

	LOG_DEBUG("%s %08X %08X", __FUNCTION__, current, (unsigned) address);

	dsp563xx_jtag_debug_request(target);

	dsp563xx_restore_context(target);

	/* reset trace mode */
	dsp563xx_once_reg_write(target->tap, DSP563XX_ONCE_OSCR, 0x000000);
	/* enable trace mode */
	dsp563xx_once_reg_write(target->tap, DSP563XX_ONCE_OSCR,
				DSP563XX_ONCE_OSCR_TME);

	cnt = 0;

	/* on JUMP we need one extra cycle */
	if (!current)
		cnt++;

	/* load step counter with N-1 */
	dsp563xx_once_reg_write(target->tap, DSP563XX_ONCE_OTC, cnt);

	if (current)
	{
		/* restore pipeline registers and go */
		dsp563xx_once_reg_write(target->tap, DSP563XX_ONCE_OPILR,
					dsp563xx->pipeline_context.once_opilr);
		dsp563xx_once_reg_write(target->tap,
					DSP563XX_ONCE_OPDBR | DSP563XX_ONCE_OCR_EX |
					DSP563XX_ONCE_OCR_GO,
					dsp563xx->pipeline_context.once_opdbr);
	}
	else
	{
		/* set to go register and jump */
		dsp563xx_once_reg_write(target->tap, DSP563XX_ONCE_OPDBR,
					DSP563XX_ASM_CMD_JUMP);
		dsp563xx_once_reg_write(target->tap,
					DSP563XX_ONCE_PDBGOTO | DSP563XX_ONCE_OCR_EX
					| DSP563XX_ONCE_OCR_GO, address);
	}

	while (1)
	{
		dsp563xx_once_reg_read(target->tap, DSP563XX_ONCE_OSCR,
				       &once_status);

		if (once_status & DSP563XX_ONCE_OSCR_TO)
		{
			/* store pipeline register */
			dsp563xx_once_reg_read(target->tap, DSP563XX_ONCE_OPILR,
					       &dsp563xx->pipeline_context.
					       once_opilr);
			dsp563xx_once_reg_read(target->tap, DSP563XX_ONCE_OPDBR,
					       &dsp563xx->pipeline_context.
					       once_opdbr);

			dsp563xx_save_context(target);

			dsp563xx_once_reg_read(target->tap, DSP563XX_ONCE_OPABFR,
					       &dr_in);
			LOG_DEBUG("%08X", (unsigned) dr_in);
			dsp563xx_once_reg_read(target->tap, DSP563XX_ONCE_OPABDR,
					       &dr_in);
			LOG_DEBUG("%08X", (unsigned) dr_in);
			dsp563xx_once_reg_read(target->tap, DSP563XX_ONCE_OPABEX,
					       &dr_in);
			LOG_DEBUG("%08X", (unsigned) dr_in);

			/* reset trace mode */
			dsp563xx_once_reg_write(target->tap, DSP563XX_ONCE_OSCR,
						0x000000);

			break;
		}
	}

	return ERROR_OK;
}

static int dsp563xx_assert_reset(struct target *target)
{
	target->state = TARGET_RESET;

	LOG_DEBUG("%s", __FUNCTION__);
	return ERROR_OK;
}

static int dsp563xx_deassert_reset(struct target *target)
{
	target->state = TARGET_RUNNING;

	LOG_DEBUG("%s", __FUNCTION__);
	return ERROR_OK;
}

static int dsp563xx_soft_reset_halt(struct target *target)
{
	LOG_DEBUG("%s", __FUNCTION__);
	return ERROR_OK;
}

/*
* 000000			nop
* 46F400 AABBCC		move              #$aabbcc,y0
* 60F400 AABBCC		move              #$aabbcc,r0
* 467000 AABBCC		move              y0,x:AABBCC
* 607000 AABBCC		move              r0,x:AABBCC

* 46E000		move              x:(r0),y0
* 4EE000		move              y:(r0),y0
* 07E086		move              p:(r0),y0

* 0450B9		move              sr,r0
* 0446BA		move              omr,y0
* 0446BC		move              ssh,y0
* 0446BD		move              ssl,y0
* 0446BE		move              la,y0
* 0446BF		move              lc,y0
* 
* 61F000 AABBCC		move              x:AABBCC,r1
* 076190		movem             r0,p:(r1)
*
*/
static int dsp563xx_read_memory_p(struct target *target, uint32_t address,
			   uint32_t size, uint32_t count, uint8_t * buffer)
{
	uint32_t i, x;
	uint32_t data;
	uint8_t *b;

	LOG_DEBUG("address: 0x%8.8" PRIx32 ", size: 0x%8.8" PRIx32 ", count: 0x%8.8"
		  PRIx32, address, size, count);

	if (target->state != TARGET_HALTED)
	{
		LOG_WARNING("target not halted");
		return ERROR_TARGET_NOT_HALTED;
	}

	x = count;

	for (i = 0; i < x; i++)
	{
		dsp563xx_once_execute_dw_ir_nq(target->tap, 0x60F400, address + i);
		dsp563xx_once_execute_sw_ir_nq(target->tap, 0x07E086);
		dsp563xx_once_execute_dw_ir_nq(target->tap, 0x467000, 0xfffffc);
		dsp563xx_execute_queue();

		dsp563xx_once_reg_read(target->tap, DSP563XX_ONCE_OGDBR, &data);

		b = buffer + 4 * i;
		if (size > 0)
			*b++ = data >> 0;
		if (size > 1)
			*b++ = data >> 8;
		if (size > 2)
			*b++ = data >> 16;
		if (size > 3)
			*b++ = 0x00;
	}

	return ERROR_OK;
}

static int dsp563xx_write_memory_p(struct target *target, uint32_t address, uint32_t size,
			    uint32_t count, uint8_t * buffer)
{
	uint32_t i, x;
	uint32_t data;
	uint8_t *b;

	LOG_DEBUG("address: 0x%8.8" PRIx32 ", size: 0x%8.8" PRIx32 ", count: 0x%8.8"
		  PRIx32 "", address, size, count);

	if (target->state != TARGET_HALTED)
	{
		LOG_WARNING("target not halted");
		return ERROR_TARGET_NOT_HALTED;
	}

	x = count;

	for (i = 0; i < x; i++)
	{
		b = buffer + 4 * i;

		data = 0;
		if (size > 0)
			data = *buffer++;
		if (size > 1)
			data |= (*buffer++) << 8;
		if (size > 2)
			data |= (*buffer++) << 16;
		if (size > 3)
			data |= (*buffer++) << 24;

//              LOG_DEBUG("%08X", data);

		dsp563xx_once_execute_dw_ir_nq(target->tap, 0x61F400, address + i);
		dsp563xx_once_execute_dw_ir_nq(target->tap, 0x60F400, data);
		dsp563xx_once_execute_sw_ir_nq(target->tap, 0x076190);
		dsp563xx_execute_queue();
	}

	return ERROR_OK;
}

static int dsp563xx_jtag_sendinstr(struct jtag_tap *tap, uint8_t * ir_in, uint8_t ir_out)
{
	return dsp563xx_write_ir_u8(tap, ir_in, ir_out, DSP563XX_JTAG_INS_LEN, 1);
}

/* IR and DR functions */
static int dsp563xx_write_ir(struct jtag_tap *tap, uint8_t * ir_in, uint8_t * ir_out,
		      int ir_len, int rti)
{
	if (NULL == tap)
	{
		LOG_ERROR("invalid tap");
		return ERROR_FAIL;
	}
	if (ir_len != tap->ir_length)
	{
		LOG_ERROR("invalid ir_len");
		return ERROR_FAIL;
	}

	{
		jtag_add_plain_ir_scan(tap->ir_length, ir_out, ir_in, TAP_IDLE);
	}

	return ERROR_OK;
}

static int dsp563xx_write_dr(struct jtag_tap *tap, uint8_t * dr_in, uint8_t * dr_out,
		      int dr_len, int rti)
{
	if (NULL == tap)
	{
		LOG_ERROR("invalid tap");
		return ERROR_FAIL;
	}

	{
		jtag_add_plain_dr_scan(dr_len, dr_out, dr_in, TAP_IDLE);
	}

	return ERROR_OK;
}

static int dsp563xx_write_ir_u8(struct jtag_tap *tap, uint8_t * ir_in, uint8_t ir_out,
			 int ir_len, int rti)
{
	if (ir_len > 8)
	{
		LOG_ERROR("ir_len overflow, maxium is 8");
		return ERROR_FAIL;
	}

	dsp563xx_write_ir(tap, ir_in, &ir_out, ir_len, rti);

	return ERROR_OK;
}

int dsp563xx_write_dr_u8(struct jtag_tap *tap, uint8_t * dr_in, uint8_t dr_out,
			 int dr_len, int rti)
{
	if (dr_len > 8)
	{
		LOG_ERROR("dr_len overflow, maxium is 8");
		return ERROR_FAIL;
	}

	dsp563xx_write_dr(tap, dr_in, &dr_out, dr_len, rti);

	return ERROR_OK;
}

int dsp563xx_write_dr_u32(struct jtag_tap *tap, uint32_t * dr_in, uint32_t dr_out,
			  int dr_len, int rti)
{
	if (dr_len > 32)
	{
		LOG_ERROR("dr_len overflow, maxium is 32");
		return ERROR_FAIL;
	}

	dsp563xx_write_dr(tap, (uint8_t *) dr_in, (uint8_t *) & dr_out, dr_len, rti);

	return ERROR_OK;
}

int dsp563xx_execute_queue(void)
{
	return jtag_execute_queue();
}

/** Holds methods for DSP563XX targets. */
struct target_type dsp563xx_target = {
	.name = "dsp563xx",

	.poll = dsp563xx_poll,
	.arch_state = dsp563xx_arch_state,

	.target_request_data = NULL,

	.get_gdb_reg_list = dsp563xx_get_gdb_reg_list,

	.halt = dsp563xx_halt,
	.resume = dsp563xx_resume,
	.step = dsp563xx_step,

	.assert_reset = dsp563xx_assert_reset,
	.deassert_reset = dsp563xx_deassert_reset,
	.soft_reset_halt = dsp563xx_soft_reset_halt,

	.read_memory = dsp563xx_read_memory_p,
	.write_memory = dsp563xx_write_memory_p,

	.target_create = dsp563xx_target_create,
	.init_target = dsp563xx_init_target,
};