| /* |
| * 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 <algorithm> |
| #include <iostream> |
| #include <typeinfo> |
| |
| #include <absl/strings/str_cat.h> |
| #include <absl/types/span.h> |
| #include <gflags/gflags.h> |
| #include <prjxray/memory_mapped_file.h> |
| #include <prjxray/xilinx/architectures.h> |
| #include <prjxray/xilinx/bitstream_reader.h> |
| |
| namespace xilinx = prjxray::xilinx; |
| |
| struct Action { |
| std::string name; |
| std::function<int(int, char* [])> handler; |
| }; |
| |
| struct ConfigPacketsLister { |
| ConfigPacketsLister(const absl::Span<uint8_t>& bytes) : bytes_(bytes) {} |
| |
| const absl::Span<uint8_t>& bytes_; |
| |
| template <typename T> |
| int operator()(T& arg) { |
| using ArchType = std::decay_t<decltype(arg)>; |
| auto reader = |
| xilinx::BitstreamReader<ArchType>::InitWithBytes(bytes_); |
| if (!reader) { |
| std::cerr << "Input doesn't look like a bitstream" |
| << std::endl; |
| return 1; |
| } |
| |
| for (auto packet : *reader) { |
| std::cout << packet; |
| } |
| |
| return 0; |
| } |
| }; |
| |
| int ListConfigPackets(int argc, char* argv[]) { |
| std::string architecture("Series7"); |
| if (argc < 1) { |
| std::cerr << "ERROR: no input specified" << std::endl; |
| std::cerr << "Usage: " << argv[0] |
| << "list_config_packets <bit_file>" << std::endl; |
| return 1; |
| } |
| if (argc == 2) { |
| architecture = argv[1]; |
| } |
| |
| auto in_file_name = argv[0]; |
| auto in_file = prjxray::MemoryMappedFile::InitWithFile(in_file_name); |
| if (!in_file) { |
| std::cerr << "Unable to open bit file: " << in_file_name |
| << std::endl; |
| return 1; |
| } |
| auto in_bytes = absl::Span<uint8_t>( |
| static_cast<uint8_t*>(in_file->data()), in_file->size()); |
| xilinx::Architecture::Container arch_container = |
| xilinx::ArchitectureFactory::create_architecture(architecture); |
| return absl::visit(ConfigPacketsLister(in_bytes), arch_container); |
| } |
| |
| struct DebugFrameAddressesDumper { |
| DebugFrameAddressesDumper(const absl::Span<uint8_t>& bytes) |
| : bytes_(bytes) {} |
| |
| const absl::Span<uint8_t>& bytes_; |
| |
| template <typename T> |
| int operator()(T& arg) { |
| using ArchType = std::decay_t<decltype(arg)>; |
| auto reader = |
| xilinx::BitstreamReader<ArchType>::InitWithBytes(bytes_); |
| if (!reader) { |
| std::cerr << "Input doesn't look like a bitstream" |
| << std::endl; |
| return 1; |
| } |
| |
| bool found_one_lout = false; |
| for (auto packet : *reader) { |
| if ((packet.opcode() != |
| xilinx::ConfigurationPacket< |
| typename ArchType::ConfRegType>::Opcode:: |
| Write) || |
| (packet.address() != ArchType::ConfRegType::LOUT)) { |
| continue; |
| } |
| |
| if (packet.data().size() != 1) { |
| std::cerr << "Write to FAR with word_count != 1" |
| << std::endl; |
| continue; |
| } |
| |
| found_one_lout = true; |
| std::cout << std::dec << packet.data()[0] << std::endl; |
| } |
| |
| if (!found_one_lout) { |
| std::cerr |
| << "No LOUT writes found. Was " |
| << "BITSTREAM.GENERAL.DEBUGBITSTREAM set to YES?" |
| << std::endl; |
| return 1; |
| } |
| |
| return 0; |
| } |
| }; |
| |
| int DumpDebugbitstreamFrameAddresses(int argc, char* argv[]) { |
| std::string architecture("Series7"); |
| if (argc < 1) { |
| std::cerr << "ERROR: no input specified" << std::endl; |
| std::cerr << "Usage: " << argv[0] |
| << "dump_debugbitstream_frame_addresses <bit_file>" |
| << std::endl; |
| return 1; |
| } |
| if (argc == 2) { |
| architecture = argv[1]; |
| } |
| |
| auto in_file_name = argv[0]; |
| auto in_file = prjxray::MemoryMappedFile::InitWithFile(in_file_name); |
| if (!in_file) { |
| std::cerr << "Unable to open bit file: " << in_file_name |
| << std::endl; |
| return 1; |
| } |
| |
| auto in_bytes = absl::Span<uint8_t>( |
| static_cast<uint8_t*>(in_file->data()), in_file->size()); |
| xilinx::Architecture::Container arch_container = |
| xilinx::ArchitectureFactory::create_architecture(architecture); |
| return absl::visit(DebugFrameAddressesDumper(in_bytes), arch_container); |
| } |
| |
| struct DeviceIdGetter { |
| DeviceIdGetter(const absl::Span<uint8_t>& bytes) : bytes_(bytes) {} |
| |
| const absl::Span<uint8_t>& bytes_; |
| |
| template <typename T> |
| int operator()(T& arg) { |
| using ArchType = std::decay_t<decltype(arg)>; |
| auto reader = |
| xilinx::BitstreamReader<ArchType>::InitWithBytes(bytes_); |
| if (!reader) { |
| std::cerr << "Input doesn't look like a bitstream" |
| << std::endl; |
| return 1; |
| } |
| auto idcode_packet = std::find_if( |
| reader->begin(), reader->end(), |
| [](const xilinx::ConfigurationPacket< |
| typename ArchType::ConfRegType>& packet) { |
| return (packet.opcode() == |
| xilinx::ConfigurationPacket< |
| typename ArchType::ConfRegType>:: |
| Opcode::Write) && |
| (packet.address() == |
| ArchType::ConfRegType::IDCODE); |
| }); |
| if (idcode_packet != reader->end()) { |
| if (std::is_same<ArchType, xilinx::Spartan6>::value) { |
| if (idcode_packet->data().size() != 2) { |
| std::cerr << "Write to IDCODE with " |
| "word_count != 2" |
| << std::endl; |
| return 1; |
| } |
| std::cout << "0x" << std::hex |
| << ((idcode_packet->data()[0] << 16) | |
| idcode_packet->data()[1]) |
| << std::endl; |
| } else { |
| if (idcode_packet->data().size() != 1) { |
| std::cerr << "Write to IDCODE with " |
| "word_count != 1" |
| << std::endl; |
| return 1; |
| } |
| std::cout << "0x" << std::hex |
| << idcode_packet->data()[0] |
| << std::endl; |
| } |
| } |
| return 0; |
| } |
| }; |
| |
| int GetDeviceId(int argc, char* argv[]) { |
| std::string architecture("Series7"); |
| if (argc < 1) { |
| std::cerr << "ERROR: no input specified" << std::endl; |
| std::cerr << "Usage: " << argv[0] |
| << "get_device_id <bit_file> [<architecture>]" |
| << std::endl; |
| return 1; |
| } |
| if (argc == 2) { |
| architecture = argv[1]; |
| } |
| |
| auto in_file_name = argv[0]; |
| auto in_file = prjxray::MemoryMappedFile::InitWithFile(in_file_name); |
| if (!in_file) { |
| std::cerr << "Unable to open bit file: " << in_file_name |
| << std::endl; |
| return 1; |
| } |
| auto in_bytes = absl::Span<uint8_t>( |
| static_cast<uint8_t*>(in_file->data()), in_file->size()); |
| |
| xilinx::Architecture::Container arch_container = |
| xilinx::ArchitectureFactory::create_architecture(architecture); |
| return absl::visit(DeviceIdGetter(in_bytes), arch_container); |
| } |
| |
| int main(int argc, char* argv[]) { |
| Action actions[] = { |
| {"list_config_packets", ListConfigPackets}, |
| {"dump_debugbitstream_frame_addresses", |
| DumpDebugbitstreamFrameAddresses}, |
| {"get_device_id", GetDeviceId}, |
| }; |
| |
| gflags::SetUsageMessage( |
| absl::StrCat("Usage: ", argv[0], " [options] [bitfile]")); |
| gflags::ParseCommandLineFlags(&argc, &argv, true); |
| |
| if (argc < 2) { |
| std::cerr << "ERROR: no command specified" << std::endl; |
| std::cerr << "Usage: " << argv[0] |
| << "<command> <command_options>" << std::endl; |
| return 1; |
| } |
| |
| auto requested_action_str = argv[1]; |
| auto requested_action = std::find_if( |
| std::begin(actions), std::end(actions), |
| [&](const Action& t) { return t.name == requested_action_str; }); |
| if (requested_action == std::end(actions)) { |
| std::cerr << "Unknown action: " << requested_action_str |
| << std::endl; |
| return 1; |
| } |
| |
| return requested_action->handler(argc - 2, argv + 2); |
| } |