couple of functions implemented
This commit is contained in:
parent
6ffbb9ed87
commit
b8f58d558c
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* nextpnr -- Next Generation Place and Route
|
* nextpnr -- Next Generation Place and Route
|
||||||
*
|
*
|
||||||
* Copyright (C) 2020 Lofty <dan.ravensloft@gmail.com
|
* Copyright (C) 2021 Lofty <dan.ravensloft@gmail.com>
|
||||||
*
|
*
|
||||||
* Permission to use, copy, modify, and/or distribute this software for any
|
* Permission to use, copy, modify, and/or distribute this software for any
|
||||||
* purpose with or without fee is hereby granted, provided that the above
|
* purpose with or without fee is hereby granted, provided that the above
|
||||||
@ -14,5 +14,82 @@
|
|||||||
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
||||||
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
||||||
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
||||||
*
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
|
#include "nextpnr.h"
|
||||||
|
|
||||||
|
#include "mistral/lib/cyclonev.h"
|
||||||
|
|
||||||
|
NEXTPNR_NAMESPACE_BEGIN
|
||||||
|
|
||||||
|
using mistral::CycloneV;
|
||||||
|
|
||||||
|
Arch::Arch(ArchArgs args)
|
||||||
|
{
|
||||||
|
this->args = args;
|
||||||
|
this->cyclonev = mistral::CycloneV::get_model(args.device);
|
||||||
|
NPNR_ASSERT(this->cyclonev != nullptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
int Arch::getTileBelDimZ(int x, int y) const
|
||||||
|
{
|
||||||
|
CycloneV::pos_t pos = cyclonev->xy2pos(x, y);
|
||||||
|
|
||||||
|
for (CycloneV::block_type_t bel : cyclonev->pos_get_bels(pos)) {
|
||||||
|
switch (bel) {
|
||||||
|
case CycloneV::block_type_t::LAB:
|
||||||
|
/*
|
||||||
|
* nextpnr and mistral disagree on what a BEL is: mistral thinks an entire LAB
|
||||||
|
* is one BEL, but nextpnr wants something with more precision.
|
||||||
|
*
|
||||||
|
* One LAB contains 10 ALMs.
|
||||||
|
* One ALM contains 2 LUT outputs and 4 flop outputs.
|
||||||
|
*/
|
||||||
|
return 60;
|
||||||
|
case CycloneV::block_type_t::GPIO:
|
||||||
|
// GPIO tiles contain 4 pins.
|
||||||
|
return 4;
|
||||||
|
default:
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// As a temporary hack, only LABs and IO are allowed to be placed, so every other tile type has zero BELs.
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
BelId Arch::getBelByName(IdString name) const
|
||||||
|
{
|
||||||
|
char bel_type_str[80] = {0};
|
||||||
|
int x = 0, y = 0, z = 0;
|
||||||
|
BelId bel;
|
||||||
|
|
||||||
|
sscanf(name.c_str(this), "%s.%d.%d.%d", bel_type_str, &x, &y, &z);
|
||||||
|
|
||||||
|
auto bel_type = cyclonev->block_type_lookup(std::string{bel_type_str});
|
||||||
|
|
||||||
|
bel.pos = CycloneV::xy2pos(x, y);
|
||||||
|
bel.z = (bel_type << 8) | z;
|
||||||
|
|
||||||
|
return bel;
|
||||||
|
}
|
||||||
|
|
||||||
|
IdString Arch::getBelName(BelId bel) const
|
||||||
|
{
|
||||||
|
char bel_str[80] = {0};
|
||||||
|
|
||||||
|
int x = CycloneV::pos2x(bel.pos);
|
||||||
|
int y = CycloneV::pos2y(bel.pos);
|
||||||
|
int z = bel.pos & 0xFF;
|
||||||
|
int bel_type = bel.z >> 8;
|
||||||
|
|
||||||
|
snprintf(bel_str, 80, "%s.%03d.%03d.%03d", cyclonev->block_type_names[bel_type], x, y, z);
|
||||||
|
|
||||||
|
return id(bel_str);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
NEXTPNR_NAMESPACE_END
|
218
cyclonev/arch.h
218
cyclonev/arch.h
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* nextpnr -- Next Generation Place and Route
|
* nextpnr -- Next Generation Place and Route
|
||||||
*
|
*
|
||||||
* Copyright (C) 2020 Lofty <dan.ravensloft@gmail.com>
|
* Copyright (C) 2021 Lofty <dan.ravensloft@gmail.com>
|
||||||
*
|
*
|
||||||
* Permission to use, copy, modify, and/or distribute this software for any
|
* Permission to use, copy, modify, and/or distribute this software for any
|
||||||
* purpose with or without fee is hereby granted, provided that the above
|
* purpose with or without fee is hereby granted, provided that the above
|
||||||
@ -21,265 +21,95 @@
|
|||||||
#error Include "arch.h" via "nextpnr.h" only.
|
#error Include "arch.h" via "nextpnr.h" only.
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include "mistral/lib/cyclonev.h"
|
||||||
|
|
||||||
NEXTPNR_NAMESPACE_BEGIN
|
NEXTPNR_NAMESPACE_BEGIN
|
||||||
|
|
||||||
struct BelIterator
|
|
||||||
{
|
|
||||||
int cursor;
|
|
||||||
|
|
||||||
BelIterator operator++()
|
|
||||||
{
|
|
||||||
cursor++;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
BelIterator operator++(int)
|
|
||||||
{
|
|
||||||
BelIterator prior(*this);
|
|
||||||
cursor++;
|
|
||||||
return prior;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator!=(const BelIterator &other) const { return cursor != other.cursor; }
|
|
||||||
|
|
||||||
bool operator==(const BelIterator &other) const { return cursor == other.cursor; }
|
|
||||||
|
|
||||||
BelId operator*() const
|
|
||||||
{
|
|
||||||
BelId ret;
|
|
||||||
ret.index = cursor;
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct BelRange
|
|
||||||
{
|
|
||||||
BelIterator b, e;
|
|
||||||
BelIterator begin() const { return b; }
|
|
||||||
BelIterator end() const { return e; }
|
|
||||||
};
|
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
|
|
||||||
struct BelPinIterator
|
|
||||||
{
|
|
||||||
const /* something */ int *ptr = nullptr;
|
|
||||||
|
|
||||||
void operator++() { ptr++; }
|
|
||||||
bool operator!=(const BelPinIterator &other) const { return ptr != other.ptr; }
|
|
||||||
|
|
||||||
BelPin operator*() const
|
|
||||||
{
|
|
||||||
BelPin ret;
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct BelPinRange
|
|
||||||
{
|
|
||||||
BelPinIterator b, e;
|
|
||||||
BelPinIterator begin() const { return b; }
|
|
||||||
BelPinIterator end() const { return e; }
|
|
||||||
};
|
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
|
|
||||||
struct WireIterator
|
|
||||||
{
|
|
||||||
int cursor = -1;
|
|
||||||
|
|
||||||
void operator++() { cursor++; }
|
|
||||||
bool operator!=(const WireIterator &other) const { return cursor != other.cursor; }
|
|
||||||
|
|
||||||
WireId operator*() const
|
|
||||||
{
|
|
||||||
WireId ret;
|
|
||||||
ret.index = cursor;
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct WireRange
|
|
||||||
{
|
|
||||||
WireIterator b, e;
|
|
||||||
WireIterator begin() const { return b; }
|
|
||||||
WireIterator end() const { return e; }
|
|
||||||
};
|
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
|
|
||||||
struct AllPipIterator
|
|
||||||
{
|
|
||||||
int cursor = -1;
|
|
||||||
|
|
||||||
void operator++() { cursor++; }
|
|
||||||
bool operator!=(const AllPipIterator &other) const { return cursor != other.cursor; }
|
|
||||||
|
|
||||||
PipId operator*() const
|
|
||||||
{
|
|
||||||
PipId ret;
|
|
||||||
ret.index = cursor;
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct AllPipRange
|
|
||||||
{
|
|
||||||
AllPipIterator b, e;
|
|
||||||
AllPipIterator begin() const { return b; }
|
|
||||||
AllPipIterator end() const { return e; }
|
|
||||||
};
|
|
||||||
|
|
||||||
// -----------------------------------------------------------------------
|
|
||||||
|
|
||||||
struct PipIterator
|
|
||||||
{
|
|
||||||
const int *cursor = nullptr;
|
|
||||||
|
|
||||||
void operator++() { cursor++; }
|
|
||||||
bool operator!=(const PipIterator &other) const { return cursor != other.cursor; }
|
|
||||||
|
|
||||||
PipId operator*() const
|
|
||||||
{
|
|
||||||
PipId ret;
|
|
||||||
ret.index = *cursor;
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct PipRange
|
|
||||||
{
|
|
||||||
PipIterator b, e;
|
|
||||||
PipIterator begin() const { return b; }
|
|
||||||
PipIterator end() const { return e; }
|
|
||||||
};
|
|
||||||
|
|
||||||
struct ArchArgs
|
struct ArchArgs
|
||||||
{
|
{
|
||||||
|
std::string device;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Arch : BaseCtx
|
struct Arch : BaseCtx
|
||||||
{
|
{
|
||||||
ArchArgs args;
|
ArchArgs args;
|
||||||
|
mistral::CycloneV* cyclonev;
|
||||||
|
|
||||||
Arch(ArchArgs args);
|
Arch(ArchArgs args);
|
||||||
|
|
||||||
std::string getChipName() const;
|
std::string getChipName() const { return std::string{"TODO: getChipName"}; }
|
||||||
|
|
||||||
IdString archId() const { return id("cyclonev"); }
|
IdString archId() const { return id("cyclonev"); }
|
||||||
ArchArgs archArgs() const;
|
ArchArgs archArgs() const { return args; }
|
||||||
IdString archArgsToId(ArchArgs args) const;
|
IdString archArgsToId(ArchArgs args) const { return id("TODO: archArgsToId"); }
|
||||||
|
|
||||||
// -------------------------------------------------
|
// -------------------------------------------------
|
||||||
|
|
||||||
int getGridDimX() const;
|
int getGridDimX() const { return cyclonev->get_tile_sx(); }
|
||||||
int getGridDimY() const;
|
int getGridDimY() const { return cyclonev->get_tile_sy(); }
|
||||||
int getTileBelDimZ(int, int) const;
|
int getTileBelDimZ(int x, int y) const; // arch.cc
|
||||||
int getTilePipDimZ(int, int) const;
|
int getTilePipDimZ(int x, int y) const { return 1; }
|
||||||
|
|
||||||
// -------------------------------------------------
|
// -------------------------------------------------
|
||||||
|
|
||||||
BelId getBelByName(IdString name) const;
|
BelId getBelByName(IdString name) const; // arch.cc
|
||||||
|
IdString getBelName(BelId bel) const; // arch.cc
|
||||||
IdString getBelName(BelId bel) const;
|
uint32_t getBelChecksum(BelId bel) const { return (bel.pos << 16) | bel.z; }
|
||||||
|
|
||||||
uint32_t getBelChecksum(BelId bel) const;
|
|
||||||
|
|
||||||
void bindBel(BelId bel, CellInfo *cell, PlaceStrength strength);
|
void bindBel(BelId bel, CellInfo *cell, PlaceStrength strength);
|
||||||
|
|
||||||
void unbindBel(BelId bel);
|
void unbindBel(BelId bel);
|
||||||
|
|
||||||
bool checkBelAvail(BelId bel) const;
|
bool checkBelAvail(BelId bel) const;
|
||||||
|
|
||||||
CellInfo *getBoundBelCell(BelId bel) const;
|
CellInfo *getBoundBelCell(BelId bel) const;
|
||||||
|
|
||||||
CellInfo *getConflictingBelCell(BelId bel) const;
|
CellInfo *getConflictingBelCell(BelId bel) const;
|
||||||
|
const std::vector<BelId> &getBels() const;
|
||||||
BelRange getBels() const;
|
|
||||||
|
|
||||||
Loc getBelLocation(BelId bel) const;
|
Loc getBelLocation(BelId bel) const;
|
||||||
|
|
||||||
BelId getBelByLocation(Loc loc) const;
|
BelId getBelByLocation(Loc loc) const;
|
||||||
BelRange getBelsByTile(int x, int y) const;
|
const std::vector<BelId> &getBelsByTile(int x, int y) const;
|
||||||
|
|
||||||
bool getBelGlobalBuf(BelId bel) const;
|
bool getBelGlobalBuf(BelId bel) const;
|
||||||
|
|
||||||
IdString getBelType(BelId bel) const;
|
IdString getBelType(BelId bel) const;
|
||||||
|
|
||||||
std::vector<std::pair<IdString, std::string>> getBelAttrs(BelId bel) const;
|
std::vector<std::pair<IdString, std::string>> getBelAttrs(BelId bel) const;
|
||||||
|
|
||||||
WireId getBelPinWire(BelId bel, IdString pin) const;
|
WireId getBelPinWire(BelId bel, IdString pin) const;
|
||||||
PortType getBelPinType(BelId bel, IdString pin) const;
|
PortType getBelPinType(BelId bel, IdString pin) const;
|
||||||
std::vector<IdString> getBelPins(BelId bel) const;
|
std::vector<IdString> getBelPins(BelId bel) const;
|
||||||
|
|
||||||
bool isBelLocked(BelId bel) const;
|
bool isBelLocked(BelId bel) const;
|
||||||
|
|
||||||
// -------------------------------------------------
|
// -------------------------------------------------
|
||||||
|
|
||||||
WireId getWireByName(IdString name) const;
|
WireId getWireByName(IdString name) const;
|
||||||
|
|
||||||
IdString getWireName(WireId wire) const;
|
IdString getWireName(WireId wire) const;
|
||||||
|
|
||||||
IdString getWireType(WireId wire) const;
|
IdString getWireType(WireId wire) const;
|
||||||
std::vector<std::pair<IdString, std::string>> getWireAttrs(WireId wire) const;
|
std::vector<std::pair<IdString, std::string>> getWireAttrs(WireId wire) const;
|
||||||
|
|
||||||
uint32_t getWireChecksum(WireId wire) const;
|
uint32_t getWireChecksum(WireId wire) const;
|
||||||
|
|
||||||
void bindWire(WireId wire, NetInfo *net, PlaceStrength strength);
|
void bindWire(WireId wire, NetInfo *net, PlaceStrength strength);
|
||||||
|
|
||||||
void unbindWire(WireId wire);
|
void unbindWire(WireId wire);
|
||||||
|
|
||||||
bool checkWireAvail(WireId wire) const;
|
bool checkWireAvail(WireId wire) const;
|
||||||
|
|
||||||
NetInfo *getBoundWireNet(WireId wire) const;
|
NetInfo *getBoundWireNet(WireId wire) const;
|
||||||
|
|
||||||
WireId getConflictingWireWire(WireId wire) const;
|
WireId getConflictingWireWire(WireId wire) const;
|
||||||
|
|
||||||
NetInfo *getConflictingWireNet(WireId wire) const;
|
NetInfo *getConflictingWireNet(WireId wire) const;
|
||||||
|
|
||||||
DelayInfo getWireDelay(WireId wire) const;
|
DelayInfo getWireDelay(WireId wire) const;
|
||||||
|
const std::vector<BelPin> &getWireBelPins(WireId wire) const;
|
||||||
BelPinRange getWireBelPins(WireId wire) const;
|
const std::vector<WireId> &getWires() const;
|
||||||
|
|
||||||
WireRange getWires() const;
|
|
||||||
|
|
||||||
// -------------------------------------------------
|
// -------------------------------------------------
|
||||||
|
|
||||||
PipId getPipByName(IdString name) const;
|
PipId getPipByName(IdString name) const;
|
||||||
|
|
||||||
void bindPip(PipId pip, NetInfo *net, PlaceStrength strength);
|
void bindPip(PipId pip, NetInfo *net, PlaceStrength strength);
|
||||||
|
|
||||||
void unbindPip(PipId pip);
|
void unbindPip(PipId pip);
|
||||||
|
|
||||||
bool checkPipAvail(PipId pip) const;
|
bool checkPipAvail(PipId pip) const;
|
||||||
|
|
||||||
NetInfo *getBoundPipNet(PipId pip) const;
|
NetInfo *getBoundPipNet(PipId pip) const;
|
||||||
|
|
||||||
WireId getConflictingPipWire(PipId pip) const;
|
WireId getConflictingPipWire(PipId pip) const;
|
||||||
|
|
||||||
NetInfo *getConflictingPipNet(PipId pip) const;
|
NetInfo *getConflictingPipNet(PipId pip) const;
|
||||||
|
const std::vector<PipId> &getPips() const;
|
||||||
AllPipRange getPips() const;
|
|
||||||
|
|
||||||
Loc getPipLocation(PipId pip) const;
|
Loc getPipLocation(PipId pip) const;
|
||||||
|
|
||||||
IdString getPipName(PipId pip) const;
|
IdString getPipName(PipId pip) const;
|
||||||
|
|
||||||
IdString getPipType(PipId pip) const;
|
IdString getPipType(PipId pip) const;
|
||||||
std::vector<std::pair<IdString, std::string>> getPipAttrs(PipId pip) const;
|
std::vector<std::pair<IdString, std::string>> getPipAttrs(PipId pip) const;
|
||||||
|
|
||||||
uint32_t getPipChecksum(PipId pip) const;
|
uint32_t getPipChecksum(PipId pip) const;
|
||||||
|
|
||||||
WireId getPipSrcWire(PipId pip) const;
|
WireId getPipSrcWire(PipId pip) const;
|
||||||
|
|
||||||
WireId getPipDstWire(PipId pip) const;
|
WireId getPipDstWire(PipId pip) const;
|
||||||
|
|
||||||
DelayInfo getPipDelay(PipId pip) const;
|
DelayInfo getPipDelay(PipId pip) const;
|
||||||
|
const std::vector<BelPin> &getPipsDownhill(WireId wire) const;
|
||||||
PipRange getPipsDownhill(WireId wire) const;
|
const std::vector<BelPin> &getPipsUphill(WireId wire) const;
|
||||||
|
const std::vector<BelPin> &getWireAliases(WireId wire) const;
|
||||||
PipRange getPipsUphill(WireId wire) const;
|
|
||||||
|
|
||||||
PipRange getWireAliases(WireId wire) const;
|
|
||||||
|
|
||||||
BelId getPackagePinBel(const std::string &pin) const;
|
BelId getPackagePinBel(const std::string &pin) const;
|
||||||
std::string getBelPackagePin(BelId bel) const;
|
std::string getBelPackagePin(BelId bel) const;
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* nextpnr -- Next Generation Place and Route
|
* nextpnr -- Next Generation Place and Route
|
||||||
*
|
*
|
||||||
* Copyright (C) 2020 Lofty <dan.ravensloft@gmail.com>
|
* Copyright (C) 2021 Lofty <dan.ravensloft@gmail.com
|
||||||
*
|
*
|
||||||
* Permission to use, copy, modify, and/or distribute this software for any
|
* Permission to use, copy, modify, and/or distribute this software for any
|
||||||
* purpose with or without fee is hereby granted, provided that the above
|
* purpose with or without fee is hereby granted, provided that the above
|
||||||
@ -21,8 +21,12 @@
|
|||||||
#error Include "archdefs.h" via "nextpnr.h" only.
|
#error Include "archdefs.h" via "nextpnr.h" only.
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include "mistral/lib/cyclonev.h"
|
||||||
|
|
||||||
NEXTPNR_NAMESPACE_BEGIN
|
NEXTPNR_NAMESPACE_BEGIN
|
||||||
|
|
||||||
|
using mistral::CycloneV;
|
||||||
|
|
||||||
typedef int delay_t;
|
typedef int delay_t;
|
||||||
|
|
||||||
struct DelayInfo
|
struct DelayInfo
|
||||||
@ -48,11 +52,13 @@ struct DelayInfo
|
|||||||
|
|
||||||
struct BelId
|
struct BelId
|
||||||
{
|
{
|
||||||
int32_t index = -1;
|
// pos_t is used for X/Y, nextpnr-cyclonev uses its own Z coordinate system.
|
||||||
|
CycloneV::pos_t pos = 0;
|
||||||
|
uint16_t z = 0;
|
||||||
|
|
||||||
bool operator==(const BelId &other) const { return index == other.index; }
|
bool operator==(const BelId &other) const { return pos == other.pos && z == other.z; }
|
||||||
bool operator!=(const BelId &other) const { return index != other.index; }
|
bool operator!=(const BelId &other) const { return pos != other.pos || z != other.z; }
|
||||||
bool operator<(const BelId &other) const { return index < other.index; }
|
bool operator<(const BelId &other) const { return pos < other.pos || (pos == other.pos && z < other.z); }
|
||||||
};
|
};
|
||||||
|
|
||||||
struct WireId
|
struct WireId
|
||||||
@ -90,10 +96,6 @@ struct DecalId
|
|||||||
enum : int8_t
|
enum : int8_t
|
||||||
{
|
{
|
||||||
TYPE_NONE,
|
TYPE_NONE,
|
||||||
TYPE_BEL,
|
|
||||||
TYPE_WIRE,
|
|
||||||
TYPE_PIP,
|
|
||||||
TYPE_GROUP
|
|
||||||
} type = TYPE_NONE;
|
} type = TYPE_NONE;
|
||||||
int32_t index = -1;
|
int32_t index = -1;
|
||||||
bool active = false;
|
bool active = false;
|
||||||
@ -108,37 +110,8 @@ struct ArchNetInfo
|
|||||||
bool is_reset = false, is_enable = false;
|
bool is_reset = false, is_enable = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct NetInfo;
|
|
||||||
|
|
||||||
struct ArchCellInfo
|
struct ArchCellInfo
|
||||||
{
|
{
|
||||||
union
|
|
||||||
{
|
|
||||||
struct
|
|
||||||
{
|
|
||||||
bool dffEnable;
|
|
||||||
bool carryEnable;
|
|
||||||
bool negClk;
|
|
||||||
int inputCount;
|
|
||||||
const NetInfo *clk, *cen, *sr;
|
|
||||||
} lcInfo;
|
|
||||||
struct
|
|
||||||
{
|
|
||||||
bool lvds;
|
|
||||||
bool global;
|
|
||||||
bool negtrig;
|
|
||||||
int pintype;
|
|
||||||
// TODO: clk packing checks...
|
|
||||||
} ioInfo;
|
|
||||||
struct
|
|
||||||
{
|
|
||||||
bool forPadIn;
|
|
||||||
} gbInfo;
|
|
||||||
struct
|
|
||||||
{
|
|
||||||
bool ledCurConnected;
|
|
||||||
} ledInfo;
|
|
||||||
};
|
|
||||||
};
|
};
|
||||||
|
|
||||||
NEXTPNR_NAMESPACE_END
|
NEXTPNR_NAMESPACE_END
|
||||||
|
Loading…
Reference in New Issue
Block a user