2022-03-24 19:04:18 +00:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
|
|
#include <cstdint>
|
|
|
|
|
#include <optional>
|
2022-03-25 00:14:32 +00:00
|
|
|
#include <vector>
|
2022-08-30 02:04:35 +01:00
|
|
|
#include <set>
|
2022-03-24 19:04:18 +00:00
|
|
|
|
2022-08-30 02:04:35 +01:00
|
|
|
#include "src/Helpers/BiMap.hpp"
|
2022-03-24 19:04:18 +00:00
|
|
|
#include "src/Targets/TargetDescriptor.hpp"
|
|
|
|
|
#include "src/Targets/TargetRegister.hpp"
|
2022-08-30 02:04:35 +01:00
|
|
|
#include "src/Targets/TargetMemory.hpp"
|
2022-03-24 19:04:18 +00:00
|
|
|
|
2022-03-25 00:19:32 +00:00
|
|
|
#include "RegisterDescriptor.hpp"
|
|
|
|
|
|
2022-03-31 16:05:39 +01:00
|
|
|
namespace Bloom::DebugServer::Gdb
|
2022-03-24 19:04:18 +00:00
|
|
|
{
|
2022-04-03 17:00:40 +01:00
|
|
|
/**
|
|
|
|
|
* GDB target descriptor.
|
|
|
|
|
*/
|
2022-08-30 02:04:35 +01:00
|
|
|
class TargetDescriptor
|
2022-03-24 19:04:18 +00:00
|
|
|
{
|
2022-08-30 02:04:35 +01:00
|
|
|
public:
|
2022-03-25 00:11:31 +00:00
|
|
|
Targets::TargetDescriptor targetDescriptor;
|
2022-03-24 19:04:18 +00:00
|
|
|
|
2022-08-30 02:04:35 +01:00
|
|
|
explicit TargetDescriptor(
|
|
|
|
|
const Targets::TargetDescriptor& targetDescriptor,
|
|
|
|
|
const BiMap<Targets::TargetMemoryType, std::uint32_t>& memoryOffsetsByType
|
|
|
|
|
)
|
2022-03-24 19:04:18 +00:00
|
|
|
: targetDescriptor(targetDescriptor)
|
2022-08-30 02:04:35 +01:00
|
|
|
, memoryOffsetsByType(memoryOffsetsByType)
|
|
|
|
|
, memoryOffsets(memoryOffsetsByType.getValues())
|
2022-03-24 19:04:18 +00:00
|
|
|
{}
|
|
|
|
|
|
2022-08-30 02:04:35 +01:00
|
|
|
virtual ~TargetDescriptor() = default;
|
|
|
|
|
|
|
|
|
|
virtual std::uint32_t getMemoryOffset(Targets::TargetMemoryType memoryType) const {
|
|
|
|
|
return this->memoryOffsetsByType.valueAt(memoryType).value_or(0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Targets::TargetMemoryType getMemoryTypeFromGdbAddress(std::uint32_t address) const {
|
|
|
|
|
// Start with the largest offset until we find a match
|
|
|
|
|
for (
|
|
|
|
|
auto memoryOffsetIt = this->memoryOffsets.rbegin();
|
|
|
|
|
memoryOffsetIt != this->memoryOffsets.rend();
|
|
|
|
|
++memoryOffsetIt
|
|
|
|
|
) {
|
|
|
|
|
if ((address & *memoryOffsetIt) != 0U) {
|
|
|
|
|
return this->memoryOffsetsByType.at(*memoryOffsetIt);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return Targets::TargetMemoryType::FLASH;
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-24 19:04:18 +00:00
|
|
|
/**
|
|
|
|
|
* Should retrieve the GDB register number, given a target register descriptor. Or std::nullopt if the target
|
|
|
|
|
* register descriptor isn't mapped to any GDB register.
|
|
|
|
|
*
|
|
|
|
|
* @param registerDescriptor
|
|
|
|
|
* @return
|
|
|
|
|
*/
|
2022-10-01 21:01:37 +01:00
|
|
|
virtual std::optional<GdbRegisterNumber> getRegisterNumberFromTargetRegisterDescriptor(
|
2022-03-24 19:04:18 +00:00
|
|
|
const Targets::TargetRegisterDescriptor& registerDescriptor
|
2022-03-25 00:12:16 +00:00
|
|
|
) const = 0;
|
2022-03-24 19:04:18 +00:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Should retrieve the GDB register descriptor for a given GDB register number.
|
|
|
|
|
*
|
|
|
|
|
* @param number
|
|
|
|
|
* @return
|
|
|
|
|
*/
|
2022-10-01 21:01:37 +01:00
|
|
|
virtual const RegisterDescriptor& getRegisterDescriptorFromNumber(GdbRegisterNumber number) const = 0;
|
2022-03-24 19:04:18 +00:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Should retrieve the mapped target register descriptor for a given GDB register number.
|
|
|
|
|
*
|
|
|
|
|
* @param number
|
|
|
|
|
* @return
|
|
|
|
|
*/
|
|
|
|
|
virtual const Targets::TargetRegisterDescriptor& getTargetRegisterDescriptorFromNumber(
|
2022-10-01 21:01:37 +01:00
|
|
|
GdbRegisterNumber number
|
2022-03-25 00:12:16 +00:00
|
|
|
) const = 0;
|
2022-03-25 00:14:32 +00:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Should return all allocated GDB register numbers for the target.
|
|
|
|
|
*
|
|
|
|
|
* @return
|
|
|
|
|
*/
|
2022-10-01 21:01:37 +01:00
|
|
|
virtual const std::vector<GdbRegisterNumber>& getRegisterNumbers() const = 0;
|
2022-08-30 02:04:35 +01:00
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
BiMap<Targets::TargetMemoryType, std::uint32_t> memoryOffsetsByType;
|
|
|
|
|
std::set<std::uint32_t> memoryOffsets;
|
2022-03-24 19:04:18 +00:00
|
|
|
};
|
|
|
|
|
}
|