| /* |
| * Copyright (C) 2017-2020 The Project X-Ray Authors. |
| * |
| * Use of this source code is governed by a ISC-style |
| * license that can be found in the LICENSE file or at |
| * https://opensource.org/licenses/ISC |
| * |
| * SPDX-License-Identifier: ISC |
| */ |
| #include <prjxray/xilinx/configuration_packet.h> |
| |
| #include <iomanip> |
| #include <iostream> |
| #include <ostream> |
| |
| #include <prjxray/bit_ops.h> |
| |
| namespace prjxray { |
| namespace xilinx { |
| |
| template <> |
| std::pair<absl::Span<uint32_t>, |
| absl::optional<ConfigurationPacket<Spartan6ConfigurationRegister>>> |
| ConfigurationPacket<Spartan6ConfigurationRegister>::InitWithWords( |
| absl::Span<uint32_t> words, |
| const ConfigurationPacket<Spartan6ConfigurationRegister>* previous_packet) { |
| using ConfigurationRegister = Spartan6ConfigurationRegister; |
| // Need at least one 32-bit word to have a valid packet header. |
| if (words.size() < 1) |
| return {words, {}}; |
| |
| uint32_t header_type = bit_field_get(words[0], 15, 13); |
| switch (header_type) { |
| case NONE: |
| // Type 0 is emitted at the end of a configuration row |
| // when BITSTREAM.GENERAL.DEBUGBITSTREAM is set to YES. |
| // These seem to be padding that are interepreted as |
| // NOPs. Since Type 0 packets don't exist according to |
| // UG470 and they seem to be zero-filled, just consume |
| // the bytes without generating a packet. |
| return {words.subspan(1), |
| {{header_type, |
| Opcode::NOP, |
| ConfigurationRegister::CRC, |
| {}}}}; |
| case TYPE1: { |
| Opcode opcode = static_cast<Opcode>( |
| bit_field_get(words[0], 12, 11)); |
| ConfigurationRegister address = |
| static_cast<ConfigurationRegister>( |
| bit_field_get(words[0], 10, 5)); |
| uint32_t data_word_count = |
| bit_field_get(words[0], 4, 0); |
| |
| // If the full packet has not been received, return as |
| // though no valid packet was found. |
| if (data_word_count > words.size() - 1) { |
| return {words, {}}; |
| } |
| |
| return {words.subspan(data_word_count + 1), |
| {{header_type, opcode, address, |
| words.subspan(1, data_word_count)}}}; |
| } |
| case TYPE2: { |
| absl::optional<ConfigurationPacket> packet; |
| Opcode opcode = static_cast<Opcode>( |
| bit_field_get(words[0], 12, 11)); |
| ConfigurationRegister address = |
| static_cast<ConfigurationRegister>( |
| bit_field_get(words[0], 10, 5)); |
| // Type 2 packets according to UG380 consist of |
| // a header word followed by 2 WCD (Word Count Data) |
| // words |
| uint32_t data_word_count = (words[1] << 16) | words[2]; |
| |
| // If the full packet has not been received, return as |
| // though no valid packet was found. |
| if (data_word_count > words.size() - 1) { |
| return {words, {}}; |
| } |
| |
| // Create a packet that contains as many data words |
| // as specified in the WCD packets, but omit them |
| // in the configuration packet along with the header |
| // FIXME Figure out why we need the extra 2 words |
| packet = ConfigurationPacket( |
| header_type, opcode, address, |
| words.subspan(3, data_word_count + 2)); |
| |
| return {words.subspan(data_word_count + 3), packet}; |
| } |
| default: |
| return {{}, {}}; |
| } |
| } |
| |
| template <> |
| std::pair<absl::Span<uint32_t>, |
| absl::optional<ConfigurationPacket<Series7ConfigurationRegister>>> |
| ConfigurationPacket<Series7ConfigurationRegister>::InitWithWords( |
| absl::Span<uint32_t> words, |
| const ConfigurationPacket<Series7ConfigurationRegister>* previous_packet) { |
| using ConfigurationRegister = Series7ConfigurationRegister; |
| // Need at least one 32-bit word to have a valid packet header. |
| if (words.size() < 1) |
| return {words, {}}; |
| |
| uint32_t header_type = bit_field_get(words[0], 31, 29); |
| switch (header_type) { |
| case NONE: |
| // Type 0 is emitted at the end of a configuration row |
| // when BITSTREAM.GENERAL.DEBUGBITSTREAM is set to YES. |
| // These seem to be padding that are interepreted as |
| // NOPs. Since Type 0 packets don't exist according to |
| // UG470 and they seem to be zero-filled, just consume |
| // the bytes without generating a packet. |
| return {words.subspan(1), |
| {{header_type, |
| Opcode::NOP, |
| ConfigurationRegister::CRC, |
| {}}}}; |
| case TYPE1: { |
| Opcode opcode = static_cast<Opcode>( |
| bit_field_get(words[0], 28, 27)); |
| ConfigurationRegister address = |
| static_cast<ConfigurationRegister>( |
| bit_field_get(words[0], 26, 13)); |
| uint32_t data_word_count = |
| bit_field_get(words[0], 10, 0); |
| |
| // If the full packet has not been received, return as |
| // though no valid packet was found. |
| if (data_word_count > words.size() - 1) { |
| return {words, {}}; |
| } |
| |
| return {words.subspan(data_word_count + 1), |
| {{header_type, opcode, address, |
| words.subspan(1, data_word_count)}}}; |
| } |
| case TYPE2: { |
| absl::optional<ConfigurationPacket> packet; |
| Opcode opcode = static_cast<Opcode>( |
| bit_field_get(words[0], 28, 27)); |
| uint32_t data_word_count = |
| bit_field_get(words[0], 26, 0); |
| |
| // If the full packet has not been received, return as |
| // though no valid packet was found. |
| if (data_word_count > words.size() - 1) { |
| return {words, {}}; |
| } |
| |
| if (previous_packet) { |
| packet = ConfigurationPacket( |
| header_type, opcode, |
| previous_packet->address(), |
| words.subspan(1, data_word_count)); |
| } |
| |
| return {words.subspan(data_word_count + 1), packet}; |
| } |
| default: |
| return {{}, {}}; |
| } |
| } |
| |
| template <class ConfigRegType> |
| std::ostream& operator<<(std::ostream& o, |
| const ConfigurationPacket<ConfigRegType>& packet) { |
| if (packet.header_type() == 0x0) { |
| return o << "[Zero-pad]" << std::endl; |
| } |
| |
| switch (packet.opcode()) { |
| case ConfigurationPacket<ConfigRegType>::Opcode::NOP: |
| o << "[NOP]" << std::endl; |
| break; |
| case ConfigurationPacket<ConfigRegType>::Opcode::Read: |
| o << "[Read Type="; |
| o << packet.header_type(); |
| o << " Address="; |
| o << std::setw(2) << std::hex; |
| o << static_cast<int>(packet.address()); |
| o << " Length="; |
| o << std::setw(10) << std::dec << packet.data().size(); |
| o << " Reg=\"" << packet.address() << "\""; |
| o << "]" << std::endl; |
| break; |
| case ConfigurationPacket<ConfigRegType>::Opcode::Write: |
| o << "[Write Type="; |
| o << packet.header_type(); |
| o << " Address="; |
| o << std::setw(2) << std::hex; |
| o << static_cast<int>(packet.address()); |
| o << " Length="; |
| o << std::setw(10) << std::dec << packet.data().size(); |
| o << " Reg=\"" << packet.address() << "\""; |
| o << "]" << std::endl; |
| o << "Data in hex:" << std::endl; |
| |
| for (size_t ii = 0; ii < packet.data().size(); ++ii) { |
| o << std::setw(8) << std::hex; |
| o << packet.data()[ii] << " "; |
| |
| if ((ii + 1) % 4 == 0) { |
| o << std::endl; |
| } |
| } |
| if (packet.data().size() % 4 != 0) { |
| o << std::endl; |
| } |
| break; |
| default: |
| o << "[Invalid Opcode]" << std::endl; |
| } |
| |
| return o; |
| } |
| |
| template std::ostream& operator<<( |
| std::ostream&, |
| const ConfigurationPacket<Spartan6ConfigurationRegister>&); |
| template std::ostream& operator<<( |
| std::ostream&, |
| const ConfigurationPacket<Series7ConfigurationRegister>&); |
| } // namespace xilinx |
| } // namespace prjxray |