|  | /* | 
|  | * 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 <functional> | 
|  |  | 
|  | #include <absl/meta/type_traits.h> | 
|  | #include <gtest/gtest.h> | 
|  | #include <prjxray/bit_ops.h> | 
|  |  | 
|  | #include <prjxray/xilinx/architectures.h> | 
|  |  | 
|  | using namespace prjxray::xilinx; | 
|  |  | 
|  | constexpr uint32_t kType1NOP = prjxray::bit_field_set<uint32_t>(0, 15, 13, 0x1); | 
|  |  | 
|  | const uint32_t MakeType1(const int opcode, | 
|  | const int address, | 
|  | const int word_count) { | 
|  | return prjxray::bit_field_set<uint32_t>( | 
|  | prjxray::bit_field_set<uint32_t>( | 
|  | prjxray::bit_field_set<uint32_t>( | 
|  | prjxray::bit_field_set<uint32_t>(0x0, 15, 13, 0x1), 12, 11, | 
|  | opcode), | 
|  | 10, 5, address), | 
|  | 4, 0, word_count); | 
|  | } | 
|  |  | 
|  | const std::vector<uint32_t> MakeType2(const int opcode, | 
|  | const int address, | 
|  | const int word_count) { | 
|  | uint32_t header = prjxray::bit_field_set<uint32_t>( | 
|  | prjxray::bit_field_set<uint32_t>( | 
|  | prjxray::bit_field_set<uint32_t>( | 
|  | prjxray::bit_field_set<uint32_t>(0x0, 15, 13, 0x2), 12, 11, | 
|  | opcode), | 
|  | 10, 5, address), | 
|  | 4, 0, 0); | 
|  | uint32_t wcr1 = (word_count >> 16) & 0xFFFF; | 
|  | uint32_t wcr2 = (word_count & 0xFFFF); | 
|  | return std::vector<uint32_t>{header, wcr1, wcr2}; | 
|  | } | 
|  |  | 
|  | TEST(ConfigPacket, InitWithZeroBytes) { | 
|  | auto packet = | 
|  | ConfigurationPacket<Spartan6::ConfRegType>::InitWithWords({}); | 
|  |  | 
|  | EXPECT_EQ(packet.first, absl::Span<uint32_t>()); | 
|  | EXPECT_FALSE(packet.second); | 
|  | } | 
|  |  | 
|  | TEST(ConfigPacket, InitWithType1Nop) { | 
|  | std::vector<uint32_t> words{kType1NOP}; | 
|  | absl::Span<uint32_t> word_span(words); | 
|  | auto packet = ConfigurationPacket<Spartan6::ConfRegType>::InitWithWords( | 
|  | word_span); | 
|  | EXPECT_EQ(packet.first, absl::Span<uint32_t>()); | 
|  | ASSERT_TRUE(packet.second); | 
|  | EXPECT_EQ(packet.second->opcode(), | 
|  | ConfigurationPacket<Spartan6::ConfRegType>::Opcode::NOP); | 
|  | EXPECT_EQ(packet.second->address(), Spartan6::ConfRegType::CRC); | 
|  | EXPECT_EQ(packet.second->data(), absl::Span<uint32_t>()); | 
|  | } | 
|  |  | 
|  | TEST(ConfigPacket, InitWithType1Read) { | 
|  | std::vector<uint32_t> words{MakeType1(0x1, 0x3, 2), 0xAA, 0xBB}; | 
|  | absl::Span<uint32_t> word_span(words); | 
|  | auto packet = ConfigurationPacket<Spartan6::ConfRegType>::InitWithWords( | 
|  | word_span); | 
|  | EXPECT_EQ(packet.first, absl::Span<uint32_t>()); | 
|  | ASSERT_TRUE(packet.second); | 
|  | EXPECT_EQ(packet.second->opcode(), | 
|  | ConfigurationPacket<Spartan6::ConfRegType>::Opcode::Read); | 
|  | EXPECT_EQ(packet.second->address(), Spartan6::ConfRegType::FDRI); | 
|  | EXPECT_EQ(packet.second->data(), word_span.subspan(1)); | 
|  | } | 
|  |  | 
|  | TEST(ConfigPacket, InitWithType1Write) { | 
|  | std::vector<uint32_t> words{MakeType1(0x2, 0x4, 2), 0xAA, 0xBB}; | 
|  | absl::Span<uint32_t> word_span(words); | 
|  | auto packet = ConfigurationPacket<Spartan6::ConfRegType>::InitWithWords( | 
|  | word_span); | 
|  | EXPECT_EQ(packet.first, absl::Span<uint32_t>()); | 
|  | ASSERT_TRUE(packet.second); | 
|  | EXPECT_EQ(packet.second->opcode(), | 
|  | ConfigurationPacket<Spartan6::ConfRegType>::Opcode::Write); | 
|  | EXPECT_EQ(packet.second->address(), Spartan6::ConfRegType::FDRO); | 
|  | EXPECT_EQ(packet.second->data(), word_span.subspan(1)); | 
|  | } | 
|  |  | 
|  | TEST(ConfigPacket, InitWithType2WithPreviousPacket) { | 
|  | std::vector<uint32_t> words{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12}; | 
|  | std::vector<uint32_t> type2 = MakeType2(0x01, 0x03, 12); | 
|  | words.insert(words.begin(), type2.begin(), type2.end()); | 
|  | absl::Span<uint32_t> word_span(words); | 
|  | auto packet = ConfigurationPacket<Spartan6::ConfRegType>::InitWithWords( | 
|  | word_span); | 
|  | EXPECT_EQ(packet.first, absl::Span<uint32_t>()); | 
|  | ASSERT_TRUE(packet.second); | 
|  | EXPECT_EQ(packet.second->opcode(), | 
|  | ConfigurationPacket<Spartan6::ConfRegType>::Opcode::Read); | 
|  | EXPECT_EQ(packet.second->address(), Spartan6::ConfRegType::FDRI); | 
|  | EXPECT_EQ(packet.second->data(), word_span.subspan(3)); | 
|  | } |