/** * see jtag_add_ir_scan() * */ int interface_jtag_add_ir_scan(struct jtag_tap* active, const struct scan_field *in_fields, tap_state_t state) { size_t num_taps = jtag_tap_count_enabled(); struct jtag_command * cmd = cmd_queue_alloc(sizeof(struct jtag_command)); struct scan_command * scan = cmd_queue_alloc(sizeof(struct scan_command)); struct scan_field * out_fields = cmd_queue_alloc(num_taps * sizeof(struct scan_field)); jtag_queue_command(cmd); cmd->type = JTAG_SCAN; cmd->cmd.scan = scan; scan->ir_scan = true; scan->num_fields = num_taps; /* one field per device */ scan->fields = out_fields; scan->end_state = state; struct scan_field * field = out_fields; /* keep track where we insert data */ /* loop over all enabled TAPs */ for (struct jtag_tap * tap = jtag_tap_next_enabled(NULL); tap != NULL; tap = jtag_tap_next_enabled(tap)) { /* search the input field list for fields for the current TAP */ if (tap == active) { /* if TAP is listed in input fields, copy the value */ tap->bypass = 0; cmd_queue_scan_field_clone(field, in_fields); } else { /* if a TAP isn't listed in input fields, set it to BYPASS */ tap->bypass = 1; field->num_bits = tap->ir_length; field->out_value = buf_set_ones(cmd_queue_alloc(DIV_ROUND_UP(tap->ir_length, 8)), tap->ir_length); field->in_value = NULL; /* do not collect input for tap's in bypass */ } /* update device information */ buf_cpy(field->out_value, tap->cur_instr, tap->ir_length); field++; } assert(field == out_fields + num_taps); /* paranoia: jtag_tap_count_enabled() and jtag_tap_next_enabled() not in sync */ return ERROR_OK; }
static int jtagspi_cmd(struct flash_bank *bank, uint8_t cmd, uint32_t *addr, uint8_t *data, int len) { struct jtagspi_flash_bank *info = bank->driver_priv; struct scan_field fields[6]; uint8_t marker = 1; uint8_t xfer_bits_buf[4]; uint8_t addr_buf[3]; uint8_t *data_buf; uint32_t xfer_bits; int is_read, lenb, n; /* LOG_DEBUG("cmd=0x%02x len=%i", cmd, len); */ is_read = (len < 0); if (is_read) len = -len; n = 0; fields[n].num_bits = 1; fields[n].out_value = ▮ fields[n].in_value = NULL; n++; xfer_bits = 8 + len - 1; /* cmd + read/write - 1 due to the counter implementation */ if (addr) xfer_bits += 24; h_u32_to_be(xfer_bits_buf, xfer_bits); flip_u8(xfer_bits_buf, xfer_bits_buf, 4); fields[n].num_bits = 32; fields[n].out_value = xfer_bits_buf; fields[n].in_value = NULL; n++; cmd = flip_u32(cmd, 8); fields[n].num_bits = 8; fields[n].out_value = &cmd; fields[n].in_value = NULL; n++; if (addr) { h_u24_to_be(addr_buf, *addr); flip_u8(addr_buf, addr_buf, 3); fields[n].num_bits = 24; fields[n].out_value = addr_buf; fields[n].in_value = NULL; n++; } lenb = DIV_ROUND_UP(len, 8); data_buf = malloc(lenb); if (lenb > 0) { if (data_buf == NULL) { LOG_ERROR("no memory for spi buffer"); return ERROR_FAIL; } if (is_read) { fields[n].num_bits = jtag_tap_count_enabled(); fields[n].out_value = NULL; fields[n].in_value = NULL; n++; fields[n].out_value = NULL; fields[n].in_value = data_buf; } else { flip_u8(data, data_buf, lenb); fields[n].out_value = data_buf; fields[n].in_value = NULL; } fields[n].num_bits = len; n++; } jtagspi_set_ir(bank); /* passing from an IR scan to SHIFT-DR clears BYPASS registers */ jtag_add_dr_scan(info->tap, n, fields, TAP_IDLE); jtag_execute_queue(); if (is_read) flip_u8(data_buf, data, lenb); free(data_buf); return ERROR_OK; }