2769 lines
90 KiB
C
2769 lines
90 KiB
C
//
|
|
// Author: Wolfgang Spraul
|
|
//
|
|
// This is free and unencumbered software released into the public domain.
|
|
// For details see the UNLICENSE file at the root of the source tree.
|
|
//
|
|
|
|
#include <stdarg.h>
|
|
#include "model.h"
|
|
#include "parts.h"
|
|
|
|
static int centx_gtp(struct fpga_model *model);
|
|
static int centy_pci_rdy(struct fpga_model *model);
|
|
static int dev_oct_calibrate(struct fpga_model *model, int y, int x, int idx);
|
|
static int dev_dna(struct fpga_model *model);
|
|
static int dev_pmv(struct fpga_model *model);
|
|
static int dev_icap(struct fpga_model *model);
|
|
static int dev_spi_access(struct fpga_model *model);
|
|
static int dev_post_crc(struct fpga_model *model);
|
|
static int dev_startup(struct fpga_model *model);
|
|
static int dev_slave_spi(struct fpga_model *model);
|
|
static int dev_suspend_sync(struct fpga_model *model);
|
|
static int centy_bram_ckpin(struct fpga_model *model);
|
|
static int pcice_sw(struct fpga_model *model);
|
|
static int term_to_io_sw(struct fpga_model *model, enum extra_wires wire);
|
|
static int init_ce_clk(struct fpga_model *model);
|
|
static int init_io(struct fpga_model *model);
|
|
static int init_routing(struct fpga_model *model);
|
|
static int init_north_south_dirwire_term(struct fpga_model *model);
|
|
static int init_east_west_dirwire_term(struct fpga_model *model);
|
|
static int init_iologic(struct fpga_model *model);
|
|
static int init_logic(struct fpga_model *model);
|
|
static int init_center(struct fpga_model *model);
|
|
static int init_hclk(struct fpga_model *model);
|
|
static int init_logicout_fw(struct fpga_model *model);
|
|
static int init_bram(struct fpga_model *model);
|
|
static int init_macc(struct fpga_model *model);
|
|
static int init_topbot_tterm_gclk(struct fpga_model *model);
|
|
static int init_bufio(struct fpga_model *model);
|
|
static int init_bscan(struct fpga_model *model);
|
|
static int init_dcm(struct fpga_model *model);
|
|
static int init_pll(struct fpga_model *model);
|
|
static int init_center_hclk(struct fpga_model *model);
|
|
static int init_center_midbuf(struct fpga_model *model);
|
|
static int init_center_reg_tblr(struct fpga_model *model);
|
|
static int init_center_topbot_cfb_dfb(struct fpga_model *model);
|
|
|
|
int init_switches(struct fpga_model *model, int routing_sw)
|
|
{
|
|
RC_CHECK(model);
|
|
|
|
centx_gtp(model);
|
|
centy_pci_rdy(model);
|
|
|
|
dev_oct_calibrate(model, TOP_FIRST_REGULAR, LEFT_IO_DEVS, /*idx*/ 0);
|
|
dev_oct_calibrate(model, TOP_FIRST_REGULAR, LEFT_IO_DEVS, /*idx*/ 1);
|
|
dev_oct_calibrate(model, model->y_height - BOT_LAST_REGULAR_O, LEFT_IO_DEVS, /*idx*/ 0);
|
|
dev_oct_calibrate(model, model->y_height - BOT_LAST_REGULAR_O, LEFT_IO_DEVS, /*idx*/ 1);
|
|
dev_oct_calibrate(model, TOP_FIRST_REGULAR+1, model->x_width-RIGHT_IO_DEVS_O, /*idx*/ 0);
|
|
dev_oct_calibrate(model, model->y_height - BOT_LAST_REGULAR_O, model->x_width-RIGHT_IO_DEVS_O, /*idx*/ 0);
|
|
|
|
dev_dna(model);
|
|
dev_pmv(model);
|
|
dev_icap(model);
|
|
dev_spi_access(model);
|
|
dev_post_crc(model);
|
|
dev_startup(model);
|
|
dev_slave_spi(model);
|
|
dev_suspend_sync(model);
|
|
centy_bram_ckpin(model);
|
|
pcice_sw(model);
|
|
term_to_io_sw(model, IOCE);
|
|
term_to_io_sw(model, IOCLK);
|
|
term_to_io_sw(model, PLLCE);
|
|
term_to_io_sw(model, PLLCLK);
|
|
|
|
if (routing_sw)
|
|
init_routing(model);
|
|
|
|
init_logic(model);
|
|
init_iologic(model);
|
|
init_north_south_dirwire_term(model);
|
|
init_east_west_dirwire_term(model);
|
|
init_ce_clk(model);
|
|
init_io(model);
|
|
init_center(model);
|
|
init_hclk(model);
|
|
init_logicout_fw(model);
|
|
init_bram(model);
|
|
init_macc(model);
|
|
init_topbot_tterm_gclk(model);
|
|
init_bufio(model);
|
|
init_bscan(model);
|
|
init_dcm(model);
|
|
init_pll(model);
|
|
init_center_hclk(model);
|
|
init_center_midbuf(model);
|
|
init_center_reg_tblr(model);
|
|
init_center_topbot_cfb_dfb(model);
|
|
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int centx_gtp(struct fpga_model *model)
|
|
{
|
|
RC_CHECK(model);
|
|
add_switch(model, TOP_INNER_ROW, model->center_x-CENTER_CMTPLL_O,
|
|
"REGT_TTERM_GTP_CLKOUTEW0",
|
|
"REGT_TTERM_ALTGTP_CLKINEAST0", /*bidir*/ 0);
|
|
add_switch(model, TOP_INNER_ROW, model->center_x-CENTER_CMTPLL_O,
|
|
"REGT_TTERM_ALTGTP_CLKOUTEW0",
|
|
"REGT_TTERM_GTP_CLKINWEST0", /*bidir*/ 0);
|
|
add_switch(model, model->y_height-BOT_INNER_ROW, model->center_x-CENTER_CMTPLL_O,
|
|
"REGB_BTERM_GTP_CLKOUTEW0",
|
|
"REGB_BTERM_ALTGTP_CLKINEAST0", /*bidir*/ 0);
|
|
add_switch(model, model->y_height-BOT_INNER_ROW, model->center_x-CENTER_CMTPLL_O,
|
|
"REGB_BTERM_ALTGTP_CLKOUTEW0",
|
|
"REGB_BTERM_GTP_CLKINWEST0", /*bidir*/ 0);
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int centy_pci_rdy(struct fpga_model *model)
|
|
{
|
|
RC_CHECK(model);
|
|
add_switch(model, model->center_y-CENTER_Y_MINUS_3, LEFT_OUTER_COL,
|
|
"LIOB_TOP_PCI_RDY0", "LIOB_PCICE_TRDY_EXT", /*bidir*/ 0);
|
|
add_switch(model, model->center_y+CENTER_Y_PLUS_1, LEFT_OUTER_COL,
|
|
"LIOB_BOT_PCI_RDY0", "LIOB_PCI_IT_RDY", /*bidir*/ 0);
|
|
add_switch(model, model->center_y-CENTER_Y_MINUS_3, model->x_width-RIGHT_OUTER_O,
|
|
"RIOB_BOT_PCI_RDY0", "RIOB_PCI_IT_RDY", /*bidir*/ 0);
|
|
add_switch(model, model->center_y+CENTER_Y_PLUS_1, model->x_width-RIGHT_OUTER_O,
|
|
"RIOB_TOP_PCI_RDY1", "RIOB_PCI_TRDY_EXT", /*bidir*/ 0);
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int dev_oct_calibrate(struct fpga_model *model, int y, int x, int idx)
|
|
{
|
|
static const int logicin_wnums[4] = {/*i0*/ 29, 32, /*i1*/ 15, 7};
|
|
|
|
RC_CHECK(model);
|
|
add_switch(model, y, x,
|
|
pf("INT_INTERFACE_LOCAL_LOGICBIN%i", logicin_wnums[idx*2]),
|
|
pf("OCT_CALIBRATE%s_SO_PINWIRE", idx ? "1" : ""),
|
|
/*bidir*/ 0);
|
|
add_switch(model, y, x,
|
|
pf("INT_INTERFACE_LOCAL_LOGICBIN%i", logicin_wnums[idx*2+1]),
|
|
pf("OCT_CALIBRATE%s_S1_PINWIRE", idx ? "1" : ""),
|
|
/*bidir*/ 0);
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int dev_dna(struct fpga_model *model)
|
|
{
|
|
static const char *pairs[] = {
|
|
"INT_INTERFACE_LOCAL_CLK0", "DNA_PORT_CLK_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN14", "DNA_PORT_TEST_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN31", "DNA_PORT_READ_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN39", "DNA_PORT_DIN_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN8", "DNA_PORT_SHIFT_PINWIRE",
|
|
"DNA_PORT_DOUT_PINWIRE", "INT_INTERFACE_LOCAL_LOGICOUT_23",
|
|
"" };
|
|
|
|
RC_CHECK(model);
|
|
add_switch_set(model, TOP_OUTER_IO, LEFT_IO_DEVS,
|
|
/*prefix*/ 0, pairs, /*inc*/ 0);
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int dev_pmv(struct fpga_model *model)
|
|
{
|
|
static const char *pairs[] = {
|
|
"INT_INTERFACE_LOCAL_LOGICBIN20", "PMV_ENABLEB_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN54", "PMV_SELECTB0_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN48", "PMV_SELECTB1_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN23", "PMV_SELECTB2_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN57", "PMV_SELECTB3_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN44", "PMV_SELECTB4_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN4", "PMV_SELECTB5_PINWIRE",
|
|
"PMV_OUT_PINWIRE", "INT_INTERFACE_LOCAL_LOGICOUT_0",
|
|
"PMV_OUT_DIV2_PINWIRE", "INT_INTERFACE_LOCAL_LOGICOUT_1",
|
|
"PMV_OUT_DIV4_PINWIRE", "INT_INTERFACE_LOCAL_LOGICOUT_2",
|
|
"" };
|
|
|
|
RC_CHECK(model);
|
|
add_switch_set(model, TOP_OUTER_IO, LEFT_IO_DEVS,
|
|
/*prefix*/ 0, pairs, /*inc*/ 0);
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int dev_icap(struct fpga_model *model)
|
|
{
|
|
static const char *pairs[] = {
|
|
"INT_INTERFACE_LOCAL_CLK1", "ICAP_CLK_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN7", "ICAP_CE_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN42", "ICAP_WRITE_PINWIRE",
|
|
"ICAP_BUSY_PINWIRE", "INT_INTERFACE_LOCAL_LOGICOUT_3",
|
|
"" };
|
|
static const int icap_in_wnums[] =
|
|
{16, 5, 12, 47, 20, 45, 36, 17, 25, 34, 54, 48, 23, 57, 44, 4};
|
|
int i;
|
|
|
|
RC_CHECK(model);
|
|
add_switch_set(model, model->y_height-BOT_OUTER_IO, model->x_width-RIGHT_IO_DEVS_O,
|
|
/*prefix*/ 0, pairs, /*inc*/ 0);
|
|
for (i = 0; i <= 15; i++) {
|
|
add_switch(model, model->y_height-BOT_OUTER_IO,
|
|
model->x_width-RIGHT_IO_DEVS_O,
|
|
pf("INT_INTERFACE_LOCAL_LOGICBIN%i", icap_in_wnums[i]),
|
|
pf("ICAP_I%i_PINWIRE", i), /*bidir*/ 0);
|
|
add_switch(model, model->y_height-BOT_OUTER_IO,
|
|
model->x_width-RIGHT_IO_DEVS_O,
|
|
pf("ICAP_O%i_PINWIRE", i),
|
|
pf("INT_INTERFACE_LOCAL_LOGICOUT_%i", 4+i),
|
|
/*bidir*/ 0);
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int dev_spi_access(struct fpga_model *model)
|
|
{
|
|
static const char *pairs[] = {
|
|
"INT_INTERFACE_LOCAL_CLK0", "SPI_ACCESS_CLK_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN24", "SPI_ACCESS_CSB_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_SR0", "SPI_ACCESS_MOSI_PINWIRE",
|
|
"SPI_ACCESS_MISO_PINWIRE", "INT_INTERFACE_LOCAL_LOGICOUT_1",
|
|
"" };
|
|
|
|
RC_CHECK(model);
|
|
add_switch_set(model, model->y_height-BOT_OUTER_IO,
|
|
model->x_width-RIGHT_IO_DEVS_O, /*prefix*/ 0, pairs, /*inc*/ 0);
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int dev_post_crc(struct fpga_model *model)
|
|
{
|
|
RC_CHECK(model);
|
|
add_switch(model, model->y_height-BOT_INNER_IO,
|
|
model->x_width-RIGHT_IO_DEVS_O,
|
|
"POST_CRC_CRCERROR_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICOUT_8",
|
|
/*bidir*/ 0);
|
|
RC_RETURN(model);
|
|
}
|
|
static int dev_startup(struct fpga_model *model)
|
|
{
|
|
static const char *pairs[] = {
|
|
"INT_INTERFACE_LOCAL_CLK1", "STARTUP_CLK_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN1", "STARTUP_KEYCLEARB_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_LOGICBIN24", "STARTUP_GTS_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_SR1", "STARTUP_GSR_PINWIRE",
|
|
"STARTUP_CFGCLK_PINWIRE", "INT_INTERFACE_LOCAL_LOGICOUT_2",
|
|
"STARTUP_CFGMCLK_PINWIRE", "INT_INTERFACE_LOCAL_LOGICOUT_1",
|
|
"STARTUP_EOS_PINWIRE", "INT_INTERFACE_LOCAL_LOGICOUT_3",
|
|
"" };
|
|
|
|
RC_CHECK(model);
|
|
add_switch_set(model, model->y_height-BOT_INNER_IO,
|
|
model->x_width-RIGHT_IO_DEVS_O, /*prefix*/ 0, pairs, /*inc*/ 0);
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int dev_slave_spi(struct fpga_model *model)
|
|
{
|
|
static const char *pairs[] = {
|
|
"INT_INTERFACE_LOCAL_LOGICBIN15", "SLAVE_SPI_CMPMISO_PINWIRE",
|
|
"SLAVE_SPI_CMPACTIVEB_PINWIRE", "INT_INTERFACE_LOCAL_LOGICOUT_7",
|
|
"SLAVE_SPI_CMPCLK_PINWIRE", "INT_INTERFACE_LOCAL_LOGICOUT_4",
|
|
"SLAVE_SPI_CMPCSB_PINWIRE", "INT_INTERFACE_LOCAL_LOGICOUT_5",
|
|
"SLAVE_SPI_CMPMOSI_PINWIRE", "INT_INTERFACE_LOCAL_LOGICOUT_6",
|
|
"" };
|
|
|
|
RC_CHECK(model);
|
|
add_switch_set(model, model->y_height-BOT_INNER_IO,
|
|
model->x_width-RIGHT_IO_DEVS_O, /*prefix*/ 0, pairs, /*inc*/ 0);
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int dev_suspend_sync(struct fpga_model *model)
|
|
{
|
|
static const char *pairs[] = {
|
|
"INT_INTERFACE_LOCAL_CLK0", "SUSPEND_SYNC_CLK_PINWIRE",
|
|
"INT_INTERFACE_LOCAL_SR0", "SUSPEND_SYNC_SACK_PINWIRE",
|
|
"SUSPEND_SYNC_SREQ_PINWIRE", "INT_INTERFACE_LOCAL_LOGICOUT_0",
|
|
"" };
|
|
|
|
RC_CHECK(model);
|
|
add_switch_set(model, model->y_height-BOT_INNER_IO,
|
|
model->x_width-RIGHT_IO_DEVS_O, /*prefix*/ 0, pairs, /*inc*/ 0);
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int centy_bram_ckpin(struct fpga_model *model)
|
|
{
|
|
int x, i;
|
|
|
|
RC_CHECK(model);
|
|
for (x = LEFT_SIDE_WIDTH; x < model->x_width-RIGHT_SIDE_WIDTH; x++) {
|
|
if (!is_atx(X_FABRIC_BRAM_COL, model, x))
|
|
continue;
|
|
for (i = 0; i <= 7; i++) {
|
|
add_switch(model, model->center_y, x,
|
|
pf("REGH_DSP_IN_CKPIN%i", i),
|
|
pf("REGH_DSP_OUT_CKPIN%i", i), /*bidir*/ 0);
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int pcice_sw(struct fpga_model *model)
|
|
{
|
|
int x;
|
|
|
|
RC_CHECK(model);
|
|
for (x = LEFT_SIDE_WIDTH; x < model->x_width-RIGHT_SIDE_WIDTH; x++) {
|
|
if (is_atx(X_FABRIC_BRAM_COL, model, x)) {
|
|
add_switch(model, TOP_INNER_ROW, x,
|
|
"BRAM_TTERM_PCICE_IN",
|
|
"BRAM_TTERM_PCICE_OUT", /*bidir*/ 0);
|
|
add_switch(model, model->y_height-BOT_INNER_ROW, x,
|
|
"BRAM_TTERM_PCICE_IN",
|
|
"BRAM_TTERM_PCICE_OUT", /*bidir*/ 0);
|
|
} else if (is_atx(X_FABRIC_MACC_COL, model, x)) {
|
|
add_switch(model, TOP_INNER_ROW, x,
|
|
"MACCSITE2_TTERM_PCICE_IN",
|
|
"MACCSITE2_TTERM_PCICE_OUT", /*bidir*/ 0);
|
|
add_switch(model, model->y_height-BOT_INNER_ROW, x,
|
|
"MACCSITE2_TTERM_PCICE_IN",
|
|
"MACCSITE2_TTERM_PCICE_OUT", /*bidir*/ 0);
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int term_to_io_x(struct fpga_model *model, enum extra_wires wire, int x)
|
|
{
|
|
const char *s1;
|
|
int last_inc, y, i;
|
|
|
|
RC_CHECK(model);
|
|
|
|
if (wire == IOCE) {
|
|
s1 = "IOCE";
|
|
last_inc = 3;
|
|
} else if (wire == IOCLK) {
|
|
s1 = "IOCLK";
|
|
last_inc = 3;
|
|
} else if (wire == PLLCE) {
|
|
s1 = "PLLCE";
|
|
last_inc = 1;
|
|
} else if (wire == PLLCLK) {
|
|
s1 = "PLLCLK";
|
|
last_inc = 1;
|
|
} else RC_FAIL(model, EINVAL);
|
|
|
|
for (y = TOP_FIRST_REGULAR; y <= model->y_height - BOT_LAST_REGULAR_O; y++) {
|
|
if (!is_aty(Y_ROW_HORIZ_AXSYMM, model, y))
|
|
continue;
|
|
// up
|
|
for (i = 1; i <= HALF_ROW; i++) {
|
|
if (is_aty((x < model->center_x) ? Y_LEFT_WIRED : Y_RIGHT_WIRED, model, y-i)) {
|
|
for (i = 0; i <= last_inc; i++) {
|
|
add_switch(model, y, x, pf("HCLK_IOIL_%s%i", s1, i),
|
|
pf("HCLK_IOIL_%s%i_UP", s1, i), /*bidir*/ 0);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
// down
|
|
for (i = 1; i <= HALF_ROW; i++) {
|
|
if (is_aty((x < model->center_x) ? Y_LEFT_WIRED : Y_RIGHT_WIRED, model, y+i)) {
|
|
for (i = 0; i <= last_inc; i++) {
|
|
add_switch(model, y, x, pf("HCLK_IOIL_%s%i", s1, i),
|
|
pf("HCLK_IOIL_%s%i_DOWN", s1, i), /*bidir*/ 0);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int term_to_io_sw(struct fpga_model *model, enum extra_wires wire)
|
|
{
|
|
RC_CHECK(model);
|
|
term_to_io_x(model, wire, LEFT_IO_DEVS);
|
|
term_to_io_x(model, wire, model->x_width - RIGHT_IO_DEVS_O);
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_logic_tile(struct fpga_model *model, int y, int x)
|
|
{
|
|
int rc, i, j, ml;
|
|
const char* xp;
|
|
|
|
RC_CHECK(model);
|
|
if (has_device_type(model, y, x, DEV_LOGIC, LOGIC_M)) {
|
|
ml = 'M';
|
|
xp = "X";
|
|
} else if (has_device_type(model, y, x, DEV_LOGIC, LOGIC_L)) {
|
|
ml = 'L';
|
|
xp = "XX";
|
|
} else RC_FAIL(model, EINVAL);
|
|
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CLEX%c_CLK0", ml), pf("%s_CLK", xp), 0 /* bidir */))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CLEX%c_CLK1", ml), pf("%c_CLK", ml), 0 /* bidir */))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CLEX%c_SR0", ml), pf("%s_SR", xp), 0 /* bidir */))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CLEX%c_SR1", ml), pf("%c_SR", ml), 0 /* bidir */))) RC_FAIL(model, rc);
|
|
for (i = X_A1; i <= X_DX; i++) {
|
|
if ((rc = add_switch(model,y, x,
|
|
pf("CLEX%c_LOGICIN_B%i", ml, i),
|
|
pf("%s_%s", xp, logicin_str(i)),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
for (i = M_A1; i <= M_WE; i++) {
|
|
if (ml == 'L' &&
|
|
(i == M_AI || i == M_BI || i == M_CI
|
|
|| i == M_DI || i == M_WE))
|
|
continue;
|
|
if ((rc = add_switch(model,y, x,
|
|
pf("CLEX%c_LOGICIN_B%i", ml, i),
|
|
pf("%c_%s", ml, logicin_str(i)),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
for (i = X_A; i <= X_DQ; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%s_%s", xp, logicout_str(i)),
|
|
pf("CLEX%c_LOGICOUT%i", ml, i),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
for (i = M_A; i <= M_DQ; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%c_%s", ml, logicout_str(i)),
|
|
pf("CLEX%c_LOGICOUT%i", ml, i),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
for (i = 'A'; i <= 'D'; i++) {
|
|
for (j = 1; j <= 6; j++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%c_%c%i", ml, i, j),
|
|
pf("%c_%c", ml, i),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%s_%c%i", xp, i, j),
|
|
pf("%s_%c", xp, i),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%c_%c", ml, i),
|
|
pf("%c_%cMUX", ml, i),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
if (ml == 'L') {
|
|
if (has_connpt(model, y, x, "XL_COUT_N")) {
|
|
if ((rc = add_switch(model, y, x,
|
|
"XL_COUT", "XL_COUT_N",
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
if ((rc = add_switch(model, y, x,
|
|
"XL_COUT", "L_DMUX", 0 /* bidir */))) RC_FAIL(model, rc);
|
|
} else {
|
|
if (has_connpt(model, y, x, "M_COUT_N")) {
|
|
if ((rc = add_switch(model, y, x,
|
|
"M_COUT", "M_COUT_N",
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
if ((rc = add_switch(model, y, x,
|
|
"M_COUT", "M_DMUX", 0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_logic(struct fpga_model *model)
|
|
{
|
|
int x, y, rc;
|
|
|
|
RC_CHECK(model);
|
|
for (x = LEFT_SIDE_WIDTH; x < model->x_width-RIGHT_SIDE_WIDTH; x++) {
|
|
if (!is_atx(X_FABRIC_LOGIC_COL|X_CENTER_LOGIC_COL, model, x))
|
|
continue;
|
|
for (y = TOP_IO_TILES; y < model->y_height - BOT_IO_TILES; y++) {
|
|
if (has_device(model, y, x, DEV_LOGIC)) {
|
|
rc = init_logic_tile(model, y, x);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_iologic_tile(struct fpga_model *model, int y, int x)
|
|
{
|
|
int i, j, rc;
|
|
const char* io_prefix, *prefix, *prefix2;
|
|
|
|
RC_CHECK(model);
|
|
if (x < LEFT_SIDE_WIDTH) {
|
|
EXIT(x != LEFT_IO_DEVS);
|
|
io_prefix = "IOI_";
|
|
prefix = "LIOI_";
|
|
prefix2 = "LIOI_IOB_";
|
|
} else if (x >= model->x_width-RIGHT_SIDE_WIDTH) {
|
|
EXIT(x != model->x_width - RIGHT_IO_DEVS_O);
|
|
io_prefix = "RIOI_";
|
|
prefix = "RIOI_";
|
|
prefix2 = "RIOI_IOB_";
|
|
} else {
|
|
if (y == TOP_OUTER_IO) {
|
|
io_prefix = "TIOI_";
|
|
prefix = "TIOI_";
|
|
prefix2 = "TIOI_OUTER_";
|
|
} else if (y == TOP_INNER_IO) {
|
|
io_prefix = "TIOI_INNER_";
|
|
prefix = "TIOI_";
|
|
prefix2 = "TIOI_INNER_";
|
|
} else if (y == model->y_height-BOT_INNER_IO) {
|
|
io_prefix = "BIOI_INNER_";
|
|
prefix = "BIOI_";
|
|
prefix2 = "BIOI_INNER_";
|
|
} else if (y == model->y_height-BOT_OUTER_IO) {
|
|
io_prefix = "TIOI_";
|
|
prefix = "BIOI_";
|
|
prefix2 = "BIOI_OUTER_";
|
|
} else
|
|
EXIT(1);
|
|
}
|
|
|
|
for (i = 0; i <= 23; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("IOI_INTER_LOGICOUT%i", i),
|
|
pf("IOI_LOGICOUT%i", i), 0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
// switches going to IOI_INTER_LOGICOUT0:15
|
|
{ static const char* logicout_src[16] = {
|
|
/* 0 */ "FABRICOUT_ILOGIC_SITE",
|
|
"Q1_ILOGIC_SITE", "Q2_ILOGIC_SITE",
|
|
"Q3_ILOGIC_SITE", "Q4_ILOGIC_SITE",
|
|
"INCDEC_ILOGIC_SITE", "VALID_ILOGIC_SITE",
|
|
/* 7 */ "FABRICOUT_ILOGIC_SITE_S",
|
|
"Q1_ILOGIC_SITE_S", "Q2_ILOGIC_SITE_S",
|
|
"Q3_ILOGIC_SITE_S", "Q4_ILOGIC_SITE_S",
|
|
/* 12 */ "", "",
|
|
/* 14 */ "BUSY_IODELAY_SITE", "BUSY_IODELAY_SITE_S" };
|
|
for (i = 0; i < sizeof(logicout_src)/sizeof(logicout_src[0]); i++) {
|
|
if (logicout_src[i][0]) {
|
|
if ((rc = add_switch(model, y, x, logicout_src[i],
|
|
pf("IOI_INTER_LOGICOUT%i", i),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
}}
|
|
// The 6 CE lines (4*IO_CE and 2*PLL_CE) can be switched
|
|
// to 4 IOCE destinations. Each IOCE line can be driven
|
|
// by 6 CE lines.
|
|
for (i = 0; i <= 3; i++) {
|
|
for (j = 0; j <= 3; j++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%sIOCE%i", io_prefix, j),
|
|
pf("IOI_CLKDIST_IOCE%i%s",i/2,i%2?"_M":"_S"),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
for (j = 0; j <= 1; j++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%sPLLCE%i", io_prefix, j),
|
|
pf("IOI_CLKDIST_IOCE%i%s",i/2,i%2?"_M":"_S"),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
// Incoming clocks and fan can be switched to intermediates
|
|
// (5 sources per intermediate), and then to the ilogic/ologic
|
|
// devices (3 sources each) or 2*CLK1 (2 sources each).
|
|
for (i = 0; i < 4; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("IOI_CLK%i", i/2),
|
|
pf("IOI_CLK%iINTER%s",i%2,i<2?"_M":"_S"),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("IOI_GFAN%i", i/2),
|
|
pf("IOI_CLK%iINTER%s",i%2,i<2?"_M":"_S"),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%sIOCLK%i", io_prefix, i),
|
|
pf("IOI_CLK%iINTER%s",i%2,i<2?"_M":"_S"),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%sPLLCLK%i", io_prefix, i/2),
|
|
pf("IOI_CLK%iINTER%s",i/2,i%2?"_M":"_S"),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
// only PLLCLK goes to CLK2 intermediate
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%sPLLCLK%i", io_prefix, i/2),
|
|
pf("IOI_CLK2INTER%s",i%2?"_S":"_M"),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
// 2 sources each for IOI_CLKDIST_CLK1_M/_S
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("IOI_CLK%iINTER%s", i%2, i<2?"_M":"_S"),
|
|
pf("IOI_CLKDIST_CLK1%s", i<2?"_M":"_S"),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
// 3 sources each:
|
|
for (i = 0; i < 6; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("IOI_CLK%iINTER%s", i%3, i<3?"_M":"_S"),
|
|
pf("IOI_CLKDIST_CLK0_ILOGIC%s", i<3?"_M":"_S"),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("IOI_CLK%iINTER%s", i%3, i<3?"_M":"_S"),
|
|
pf("IOI_CLKDIST_CLK0_OLOGIC%s", i<3?"_M":"_S"),
|
|
0 /* bidir */))) RC_FAIL(model, rc);
|
|
}
|
|
// logicin wires
|
|
{
|
|
static const char* iologic_logicin[] =
|
|
{
|
|
[X_A3] = "CAL_IODELAY_SITE",
|
|
[X_A4] = "CAL_IODELAY_SITE_S",
|
|
[X_A6] = "CE_IODELAY_SITE_S",
|
|
[X_B1] = "INC_IODELAY_SITE_S",
|
|
[X_B2] = "TRAIN_OLOGIC_SITE",
|
|
[X_B3] = "TCE_OLOGIC_SITE_S",
|
|
[X_B6] = "T3_OLOGIC_SITE_S",
|
|
[X_C1] = "REV_OLOGIC_SITE_S",
|
|
[X_C2] = "D1_OLOGIC_SITE_S",
|
|
[X_C3] = "D2_OLOGIC_SITE_S",
|
|
[X_C4] = "D3_OLOGIC_SITE_S",
|
|
[X_C6] = "BITSLIP_ILOGIC_SITE_S",
|
|
[X_CE] = "SR_ILOGIC_SITE_S",
|
|
[X_D2] = "TCE_OLOGIC_SITE",
|
|
[X_D3] = "T1_OLOGIC_SITE",
|
|
[X_D4] = "T2_OLOGIC_SITE",
|
|
[X_D5] = "T3_OLOGIC_SITE",
|
|
[X_D6] = "T4_OLOGIC_SITE",
|
|
[X_DX] = "TRAIN_OLOGIC_SITE_S",
|
|
|
|
[M_A1] = "REV_OLOGIC_SITE",
|
|
[M_A2] = "OCE_OLOGIC_SITE",
|
|
[M_A3] = "D1_OLOGIC_SITE",
|
|
[M_A4] = "D2_OLOGIC_SITE",
|
|
[M_A6] = "D4_OLOGIC_SITE",
|
|
[M_AI] = "SR_ILOGIC_SITE",
|
|
[M_B1] = "REV_ILOGIC_SITE",
|
|
[M_B2] = "CE0_ILOGIC_SITE",
|
|
[M_B3] = "OCE_OLOGIC_SITE_S",
|
|
[M_B5] = "RST_IODELAY_SITE_S",
|
|
[M_B6] = "T2_OLOGIC_SITE_S",
|
|
[M_BI] = "D3_OLOGIC_SITE",
|
|
[M_C1] = "T1_OLOGIC_SITE_S",
|
|
[M_C3] = "CE_IODELAY_SITE",
|
|
[M_C4] = "D4_OLOGIC_SITE_S",
|
|
[M_D1] = "T4_OLOGIC_SITE_S",
|
|
[M_D2] = "RST_IODELAY_SITE",
|
|
[M_D4] = "BITSLIP_ILOGIC_SITE",
|
|
[M_D5] = "INC_IODELAY_SITE",
|
|
[M_D6] = "REV_ILOGIC_SITE_S",
|
|
[M_WE] = "CE0_ILOGIC_SITE_S",
|
|
};
|
|
for (i = 0; i < sizeof(iologic_logicin)/sizeof(*iologic_logicin); i++) {
|
|
if (!iologic_logicin[i]) continue;
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("IOI_LOGICINB%i", i),
|
|
iologic_logicin[i], /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
// GND
|
|
{
|
|
static const char* s[] = { "REV_OLOGIC_SITE",
|
|
"SR_OLOGIC_SITE", "TRAIN_OLOGIC_SITE" };
|
|
for (i = 0; i < 6; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%sGND_TIEOFF", prefix),
|
|
pf("%s%s", s[i/2], i%2 ? "" : "_S"),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
// VCC
|
|
{
|
|
static const char* s[] = { "IOCE_ILOGIC_SITE",
|
|
"IOCE_OLOGIC_SITE" };
|
|
for (i = 0; i < 4; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%sVCC_TIEOFF", prefix),
|
|
pf("%s%s", s[i/2], i%2 ? "" : "_S"),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
// CLK
|
|
{
|
|
static const char* s[] = { "CLKDIV_ILOGIC_SITE",
|
|
"CLKDIV_OLOGIC_SITE", "CLK_IODELAY_SITE" };
|
|
for (i = 0; i < 6; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("IOI_CLK%i", !(i%2)),
|
|
pf("%s%s", s[i/2], i%2 ? "" : "_S"),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
for (i = 0; i < 4; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CLK%i_ILOGIC_SITE%s", i/2, i%2 ? "_S" : ""),
|
|
pf("CFB%i_ILOGIC_SITE%s", i/2, i%2 ? "_S" : ""),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
// SR
|
|
{
|
|
static const char* s[] = { "SR_ILOGIC_SITE",
|
|
"SR_OLOGIC_SITE" };
|
|
for (i = 0; i < 4; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("IOI_SR%i", !(i%2)),
|
|
pf("%s%s", s[i/2], i%2 ? "" : "_S"),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
// IOCLK
|
|
{
|
|
for (i = 0; i < 4; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%sIOCLK%i", io_prefix, i),
|
|
pf("IOI_CLK%iINTER%s", i%2, (i/2)?"_M":"_S"),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
{
|
|
const char* pairs[] = {
|
|
"D1_OLOGIC_SITE", "OQ_OLOGIC_SITE",
|
|
"DATAOUT_IODELAY_SITE", "DDLY_ILOGIC_SITE",
|
|
"DDLY2_ILOGIC_SITE", "FABRICOUT_ILOGIC_SITE",
|
|
"DDLY_ILOGIC_SITE", "DFB_ILOGIC_SITE",
|
|
"D_ILOGIC_IDATAIN_IODELAY", "D_ILOGIC_SITE",
|
|
"D_ILOGIC_IDATAIN_IODELAY", "IDATAIN_IODELAY_SITE",
|
|
"D_ILOGIC_SITE", "DFB_ILOGIC_SITE",
|
|
"D_ILOGIC_SITE", "FABRICOUT_ILOGIC_SITE",
|
|
"T1_OLOGIC_SITE", "TQ_OLOGIC_SITE",
|
|
"TQ_OLOGIC_SITE", "TFB_ILOGIC_SITE",
|
|
"TQ_OLOGIC_SITE", "T_IODELAY_SITE",
|
|
"OQ_OLOGIC_SITE", "ODATAIN_IODELAY_SITE",
|
|
"OQ_OLOGIC_SITE", "OFB_ILOGIC_SITE" };
|
|
for (i = 0; i < sizeof(pairs)/sizeof(*pairs)/2; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pairs[i*2],
|
|
pairs[i*2+1],
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%s%s", pairs[i*2], "_S"),
|
|
pf("%s%s", pairs[i*2+1], "_S"),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
if ((rc = add_switch(model, y, x,
|
|
"DATAOUT2_IODELAY_SITE", "DDLY2_ILOGIC_SITE",
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
"DATAOUT2_IODELAY2_SITE_S", "DDLY2_ILOGIC_SITE_S",
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
for (i = 0; i < 2; i++) {
|
|
if ((rc = add_switch(model, y, x, "IOI_PCI_CE",
|
|
pf("OCE_OLOGIC_SITE%s", i?"_S":""),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
for (i = 0; i < 3; i++) {
|
|
// 3 because IBUF1 cannot be switched to non-_S
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("%sIBUF%i", prefix2, i/2),
|
|
pf("D_ILOGIC_IDATAIN_IODELAY%s", !(i%2)?"_S":""),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
{
|
|
const char* pairs[] = {
|
|
"DOUT_IODELAY_SITE%s", "%sO%i",
|
|
"OQ_OLOGIC_SITE%s", "%sO%i",
|
|
"TOUT_IODELAY_SITE%s", "%sT%i",
|
|
"TQ_OLOGIC_SITE%s", "%sT%i" };
|
|
for (i = 0; i < 8; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf(pairs[(i/2)*2], i%2?"_S":""),
|
|
pf(pairs[(i/2)*2+1], prefix2, i%2),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
{
|
|
const char* pairs[] = {
|
|
"SHIFTOUT1_OLOGIC_SITE", "SHIFTIN1_OLOGIC_SITE_S",
|
|
"SHIFTOUT2_OLOGIC_SITE", "SHIFTIN2_OLOGIC_SITE_S",
|
|
"SHIFTOUT3_OLOGIC_SITE_S", "SHIFTIN3_OLOGIC_SITE",
|
|
"SHIFTOUT4_OLOGIC_SITE_S", "SHIFTIN4_OLOGIC_SITE",
|
|
"SHIFTOUT_ILOGIC_SITE", "SHIFTIN_ILOGIC_SITE_S",
|
|
"SHIFTOUT_ILOGIC_SITE_S", "SHIFTIN_ILOGIC_SITE" };
|
|
for (i = 0; i < sizeof(pairs)/sizeof(*pairs)/2; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pairs[i*2], pairs[i*2+1],
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
{
|
|
const char* pairs[] = {
|
|
"IOI_CLKDIST_CLK0_ILOGIC%s", "CLK0_ILOGIC_SITE%s",
|
|
"IOI_CLKDIST_CLK0_ILOGIC%s", "IOCLK_IODELAY_SITE%s",
|
|
"IOI_CLKDIST_CLK0_OLOGIC%s", "CLK0_OLOGIC_SITE%s",
|
|
"IOI_CLKDIST_CLK0_OLOGIC%s", "IOCLK_IODELAY_SITE%s",
|
|
"IOI_CLKDIST_CLK1%s", "CLK1_ILOGIC_SITE%s",
|
|
"IOI_CLKDIST_CLK1%s", "CLK1_OLOGIC_SITE%s",
|
|
"IOI_CLKDIST_CLK1%s", "IOCLK1_IODELAY_SITE%s",
|
|
"IOI_CLKDIST_IOCE0%s", "IOCE_ILOGIC_SITE%s",
|
|
"IOI_CLKDIST_IOCE1%s", "IOCE_OLOGIC_SITE%s" };
|
|
for (i = 0; i < sizeof(pairs)/sizeof(*pairs); i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf(pairs[(i/2)*2], i%2?"_S":"_M"),
|
|
pf(pairs[(i/2)*2+1], i%2?"_S":""),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
{
|
|
const char* pairs[] = {
|
|
"IOI_MCB_DRPADD", "CAL_IODELAY_SITE%s",
|
|
"IOI_MCB_DRPBROADCAST", "RST_IODELAY_SITE%s",
|
|
"IOI_MCB_DRPCLK", "CLK_IODELAY_SITE%s",
|
|
"IOI_MCB_DRPCS", "INC_IODELAY_SITE%s",
|
|
"IOI_MCB_DRPSDO", "CE_IODELAY_SITE%s",
|
|
"IOI_MCB_DRPTRAIN", "TRAIN_OLOGIC_SITE%s" };
|
|
for (i = 0; i < sizeof(pairs)/sizeof(*pairs); i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pairs[(i/2)*2],
|
|
pf(pairs[(i/2)*2+1], i%2?"_S":""),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
{
|
|
const char* pairs[] = {
|
|
"IOI_MCB_OUTN_M", "D2_OLOGIC_SITE",
|
|
"IOI_MCB_OUTN_S", "D2_OLOGIC_SITE_S",
|
|
"IOI_MCB_OUTP_M", "D1_OLOGIC_SITE",
|
|
"IOI_MCB_OUTP_S", "D1_OLOGIC_SITE_S",
|
|
"IOI_MCB_DQIEN_M", "T2_OLOGIC_SITE",
|
|
"IOI_MCB_DQIEN_M", "T2_OLOGIC_SITE_S",
|
|
"IOI_MCB_DQIEN_S", "T1_OLOGIC_SITE",
|
|
"IOI_MCB_DQIEN_S", "T1_OLOGIC_SITE_S",
|
|
"FABRICOUT_ILOGIC_SITE", "IOI_MCB_INBYP_M",
|
|
"FABRICOUT_ILOGIC_SITE_S", "IOI_MCB_INBYP_S",
|
|
"OUTP_IODELAY_SITE", "IOI_MCB_IN_M",
|
|
"STUB_OUTP_IODELAY_S", "IOI_MCB_IN_S" };
|
|
for (i = 0; i < sizeof(pairs)/sizeof(*pairs)/2; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pairs[i*2], pairs[i*2+1],
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
if (x < LEFT_SIDE_WIDTH
|
|
|| x >= model->x_width-RIGHT_SIDE_WIDTH) {
|
|
if ((rc = add_switch(model, y, x,
|
|
"AUXSDOIN_IODELAY_M", "AUXSDO_IODELAY_M",
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
"AUXSDOIN_IODELAY_S", "AUXSDO_IODELAY_S",
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
} else {
|
|
if ((rc = add_switch(model, y, x,
|
|
"AUXSDOIN_IODELAY_S_STUB", "AUXSDO_IODELAY_S_STUB",
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
"AUXSDOIN_IODELAY_STUB", "AUXSDO_IODELAY_STUB",
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_iologic(struct fpga_model *model)
|
|
{
|
|
int x, y, rc;
|
|
|
|
RC_CHECK(model);
|
|
for (x = LEFT_SIDE_WIDTH; x < model->x_width - RIGHT_SIDE_WIDTH; x++) {
|
|
if (has_device(model, TOP_OUTER_IO, x, DEV_ILOGIC)) {
|
|
if ((rc = init_iologic_tile(model,
|
|
TOP_OUTER_IO, x))) RC_FAIL(model, rc);
|
|
}
|
|
if (has_device(model, TOP_INNER_IO, x, DEV_ILOGIC)) {
|
|
if ((rc = init_iologic_tile(model,
|
|
TOP_INNER_IO, x))) RC_FAIL(model, rc);
|
|
}
|
|
if (has_device(model, model->y_height-BOT_INNER_IO, x, DEV_ILOGIC)) {
|
|
if ((rc = init_iologic_tile(model,
|
|
model->y_height-BOT_INNER_IO, x))) RC_FAIL(model, rc);
|
|
}
|
|
if (has_device(model, model->y_height-BOT_OUTER_IO, x, DEV_ILOGIC)) {
|
|
if ((rc = init_iologic_tile(model,
|
|
model->y_height-BOT_OUTER_IO, x))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
for (y = TOP_IO_TILES; y < model->y_height - BOT_IO_TILES; y++) {
|
|
if (has_device(model, y, LEFT_IO_DEVS, DEV_ILOGIC)) {
|
|
if ((rc = init_iologic_tile(model,
|
|
y, LEFT_IO_DEVS))) RC_FAIL(model, rc);
|
|
}
|
|
if (has_device(model, y, model->x_width-RIGHT_IO_DEVS_O, DEV_ILOGIC)) {
|
|
if ((rc = init_iologic_tile(model,
|
|
y, model->x_width-RIGHT_IO_DEVS_O))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_north_south_dirwire_term(struct fpga_model *model)
|
|
{
|
|
static const int logicin_pairs[] = {21,20, 28,36, 52,44, 60,62};
|
|
int x, i, rc;
|
|
|
|
RC_CHECK(model);
|
|
for (x = 0; x < model->x_width; x++) {
|
|
if (!is_atx(X_ROUTING_COL, model, x))
|
|
continue;
|
|
|
|
// top
|
|
for (i = 0; i < 4; i++) {
|
|
rc = add_switch(model,
|
|
TOP_INNER_ROW, x,
|
|
pf("IOI_TTERM_LOGICIN%i", logicin_pairs[i*2]),
|
|
pf("IOI_TTERM_LOGICIN_S%i", logicin_pairs[i*2+1]),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
{ const char* s0_switches[] = {
|
|
"ER1E3", "EL1E_S0",
|
|
"SR1E_N3", "NL1E_S0",
|
|
"SS2E_N3", "NN2E_S0",
|
|
"SS4E3", "NW4E_S0",
|
|
"SW2E3", "NE2E_S0",
|
|
"SW4E3", "WW4E_S0",
|
|
"WL1E3", "WR1E_S0",
|
|
"WW2E3", "NW2E_S0", "" };
|
|
if ((rc = add_switch_set(model, TOP_INNER_ROW, x,
|
|
"IOI_TTERM_", s0_switches, /*inc*/ 0))) RC_FAIL(model, rc); }
|
|
{ const char* dir[] = {
|
|
"NN4B", "SS4A", "NN4A", "SS4M", "NN4M", "SS4C", "NN4C", "SS4E",
|
|
"NN2B", "SS2M", "NN2M", "SS2E",
|
|
"NE4B", "SE4A", "NE4A", "SE4M",
|
|
"NE2B", "SE2M",
|
|
"NW4B", "SW4A", "NW4A", "SW4M", "NW2B", "SW2M",
|
|
"NL1B", "SL1E",
|
|
"NR1B", "SR1E", "" };
|
|
if ((rc = add_switch_set(model, TOP_INNER_ROW, x,
|
|
"IOI_TTERM_", dir, /*inc*/ 3))) RC_FAIL(model, rc); }
|
|
|
|
// bottom
|
|
if (is_atx(X_FABRIC_BRAM_ROUTING_COL, model, x))
|
|
continue;
|
|
for (i = 0; i < 4; i++) {
|
|
rc = add_switch(model,
|
|
model->y_height-BOT_INNER_ROW, x,
|
|
pf("IOI_BTERM_LOGICIN%i", logicin_pairs[i*2+1]),
|
|
pf("IOI_BTERM_LOGICIN_N%i", logicin_pairs[i*2]),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
{ const char* n3_switches[] = {
|
|
"EL1E0", "ER1E_N3",
|
|
"NE2E0", "SW2E_N3",
|
|
"NL1E_S0", "SR1E_N3",
|
|
"NN2E_S0", "SS2E_N3",
|
|
"NW2E0", "WW2E_N3",
|
|
"NW4E0", "SS4E_N3",
|
|
"WR1E0", "WL1E_N3",
|
|
"WW4E0", "SW4E_N3", "" };
|
|
if ((rc = add_switch_set(model, model->y_height-BOT_INNER_ROW,
|
|
x, "IOI_BTERM_", n3_switches, /*inc*/ 0))) RC_FAIL(model, rc); }
|
|
{ const char* dir[] = {
|
|
"SS4B", "NN4A", "SS4A", "NN4M", "SS4M", "NN4C", "SS4C", "NN4E",
|
|
"SS2B", "NN2M", "SS2M", "NN2E",
|
|
"SE4B", "NE4A", "SE4A", "NE4M",
|
|
"SE2B", "NE2M",
|
|
"SW4B", "NW4A", "SW4A", "NW4M", "SW2B", "NW2M",
|
|
"NL1E", "SL1B",
|
|
"SR1B", "NR1E", "" };
|
|
if ((rc = add_switch_set(model, model->y_height-BOT_INNER_ROW,
|
|
x, "IOI_BTERM_", dir, /*inc*/ 3))) RC_FAIL(model, rc); }
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_east_west_dirwire_term(struct fpga_model *model)
|
|
{
|
|
int y, rc;
|
|
|
|
RC_CHECK(model);
|
|
for (y = TOP_IO_TILES; y < model->y_height-BOT_IO_TILES; y++) {
|
|
if (is_aty(Y_ROW_HORIZ_AXSYMM|Y_CHIP_HORIZ_REGS, model, y))
|
|
continue;
|
|
// left
|
|
{ const char* s[] = {
|
|
"NE4C", "NW4M", "SW4C", "SE4E", "SW4M", "SE4C",
|
|
"WW4A", "EE4M", "WW4B", "EE4A", "WW4C", "EE4E", "WW4M", "EE4C",
|
|
"NW4C", "NE4E",
|
|
"WW2B", "EE2M", "WW2M", "EE2E", "NW2M", "NE2E",
|
|
"WL1B", "EL1E", "WR1B", "ER1E", "" };
|
|
if ((rc = add_switch_set(model, y, LEFT_INNER_COL,
|
|
"LTERM_", s, /*inc*/ 3))) RC_FAIL(model, rc); }
|
|
{ const char* s[] = { "SW2M", "SE2E", "" };
|
|
if ((rc = add_switch_set(model, y, LEFT_INNER_COL,
|
|
"LTERM_", s, /*inc*/ 2))) RC_FAIL(model, rc); }
|
|
rc = add_switch(model, y, LEFT_INNER_COL,
|
|
"LTERM_SW2M3", "LTERM_SE2M3", 0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
|
|
// right
|
|
{ const char* s[] = {
|
|
"EE4A", "WW4M", "EE4B", "WW4A", "EE4C", "WW4E", "EE4M", "WW4C",
|
|
"SE4C", "SW4E", "SE4M", "SW4C",
|
|
"NE4C", "NW4E", "NE4M", "NW4C",
|
|
"NE2M", "NW2E", "EE2B", "WW2M", "EE2M", "WW2E", "SE2M", "SW2E",
|
|
"EL1B", "WL1E", "ER1B", "WR1E", "" };
|
|
if ((rc = add_switch_set(model, y, model->x_width-RIGHT_INNER_O,
|
|
"RTERM_", s, /*inc*/ 3))) RC_FAIL(model, rc); }
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_ce_clk(struct fpga_model *model)
|
|
{
|
|
int x, y, i, rc;
|
|
|
|
RC_CHECK(model);
|
|
// There are CE and CLK wires for IO and PLL that are going
|
|
// horizontally through the HCLK and vertically through the logic
|
|
// dev columns (except no-io).
|
|
// The following sets up their corresponding switches in the term
|
|
// tiles.
|
|
for (y = TOP_IO_TILES; y < model->y_height - BOT_IO_TILES; y++) {
|
|
if (is_aty(Y_ROW_HORIZ_AXSYMM, model, y)) {
|
|
// left
|
|
for (i = 0; i <= 3; i++) {
|
|
rc = add_switch(model, y, LEFT_INNER_COL,
|
|
pf("HCLK_IOI_LTERM_IOCE%i", i),
|
|
pf("HCLK_IOI_LTERM_IOCE%i_E", i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, y, LEFT_INNER_COL,
|
|
pf("HCLK_IOI_LTERM_IOCLK%i", i),
|
|
pf("HCLK_IOI_LTERM_IOCLK%i_E", i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
for (i = 0; i <= 1; i++) {
|
|
rc = add_switch(model, y, LEFT_INNER_COL,
|
|
pf("HCLK_IOI_LTERM_PLLCE%i", i),
|
|
pf("HCLK_IOI_LTERM_PLLCE%i_E", i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, y, LEFT_INNER_COL,
|
|
pf("HCLK_IOI_LTERM_PLLCLK%i", i),
|
|
pf("HCLK_IOI_LTERM_PLLCLK%i_E", i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
// right
|
|
for (i = 0; i <= 3; i++) {
|
|
rc = add_switch(model, y, model->x_width-RIGHT_INNER_O,
|
|
pf("HCLK_IOI_RTERM_IOCE%i", i),
|
|
pf("HCLK_IOI_RTERM_IOCE%i_W", 3-i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, y, model->x_width-RIGHT_INNER_O,
|
|
pf("HCLK_IOI_RTERM_IOCLK%i", i),
|
|
pf("HCLK_IOI_RTERM_IOCLK%i_W", 3-i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
for (i = 0; i <= 1; i++) {
|
|
rc = add_switch(model, y, model->x_width-RIGHT_INNER_O,
|
|
pf("HCLK_IOI_RTERM_PLLCEOUT%i", i),
|
|
pf("HCLK_IOI_RTERM_PLLCEOUT%i_W", i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, y, model->x_width-RIGHT_INNER_O,
|
|
pf("HCLK_IOI_RTERM_PLLCLKOUT%i", i),
|
|
pf("HCLK_IOI_RTERM_PLLCLKOUT%i_W", i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
}
|
|
for (x = LEFT_SIDE_WIDTH; x < model->x_width - RIGHT_SIDE_WIDTH; x++) {
|
|
if (is_atx(X_FABRIC_LOGIC_COL|X_CENTER_LOGIC_COL, model, x)
|
|
&& !is_atx(X_ROUTING_NO_IO, model, x-1)) {
|
|
// top
|
|
for (i = 0; i <= 3; i++) {
|
|
rc = add_switch(model, TOP_INNER_ROW, x,
|
|
pf("TTERM_CLB_IOCE%i", i),
|
|
pf("TTERM_CLB_IOCE%i_S", i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, TOP_INNER_ROW, x,
|
|
pf("TTERM_CLB_IOCLK%i", i),
|
|
pf("TTERM_CLB_IOCLK%i_S", i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
for (i = 0; i <= 1; i++) {
|
|
rc = add_switch(model, TOP_INNER_ROW, x,
|
|
pf("TTERM_CLB_PLLCE%i", i),
|
|
pf("TTERM_CLB_PLLCE%i_S", i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, TOP_INNER_ROW, x,
|
|
pf("TTERM_CLB_PLLCLK%i", i),
|
|
pf("TTERM_CLB_PLLCLK%i_S", i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
rc = add_switch(model, TOP_INNER_ROW, x,
|
|
"TTERM_CLB_PCICE",
|
|
"TTERM_CLB_PCICE_S",
|
|
0 /* bidir */);
|
|
|
|
// bottom
|
|
if (rc) RC_FAIL(model, rc);
|
|
for (i = 0; i <= 3; i++) {
|
|
rc = add_switch(model, model->y_height - BOT_INNER_ROW, x,
|
|
pf("BTERM_CLB_CEOUT%i", i),
|
|
pf("BTERM_CLB_CEOUT%i_N", i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, model->y_height - BOT_INNER_ROW, x,
|
|
pf("BTERM_CLB_CLKOUT%i", i),
|
|
pf("BTERM_CLB_CLKOUT%i_N", i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
for (i = 0; i <= 1; i++) {
|
|
rc = add_switch(model, model->y_height - BOT_INNER_ROW, x,
|
|
pf("BTERM_CLB_PLLCEOUT%i", i),
|
|
pf("BTERM_CLB_PLLCEOUT%i_N", i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, model->y_height - BOT_INNER_ROW, x,
|
|
pf("BTERM_CLB_PLLCLKOUT%i", i),
|
|
pf("BTERM_CLB_PLLCLKOUT%i_N", i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
rc = add_switch(model, model->y_height - BOT_INNER_ROW, x,
|
|
"BTERM_CLB_PCICE",
|
|
"BTERM_CLB_PCICE_N",
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_io_tile(struct fpga_model *model, int y, int x)
|
|
{
|
|
const char* prefix;
|
|
int i, num_devs, rc;
|
|
|
|
RC_CHECK(model);
|
|
if (!y) {
|
|
prefix = "TIOB";
|
|
rc = add_switch(model, y, x,
|
|
pf("%s_DIFFO_OUT2", prefix),
|
|
pf("%s_DIFFO_IN3", prefix), 0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
num_devs = 2;
|
|
} else if (y == model->y_height - BOT_OUTER_ROW) {
|
|
prefix = "BIOB";
|
|
rc = add_switch(model, y, x,
|
|
pf("%s_DIFFO_OUT3", prefix),
|
|
pf("%s_DIFFO_IN2", prefix), 0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
num_devs = 2;
|
|
} else if (!x) {
|
|
prefix = "LIOB";
|
|
num_devs = 1;
|
|
} else if (x == model->x_width - RIGHT_OUTER_O) {
|
|
prefix = "RIOB";
|
|
num_devs = 1;
|
|
} else
|
|
EXIT(1);
|
|
|
|
for (i = 0; i < num_devs*2; i++) {
|
|
rc = add_switch(model, y, x,
|
|
pf("%s_IBUF%i_PINW", prefix, i),
|
|
pf("%s_IBUF%i", prefix, i), 0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, y, x,
|
|
pf("%s_O%i", prefix, i),
|
|
pf("%s_O%i_PINW", prefix, i), 0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, y, x,
|
|
pf("%s_T%i", prefix, i),
|
|
pf("%s_T%i_PINW", prefix, i), 0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
rc = add_switch(model, y, x,
|
|
pf("%s_DIFFO_OUT0", prefix),
|
|
pf("%s_DIFFO_IN1", prefix), 0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
for (i = 0; i <= 1; i++) {
|
|
rc = add_switch(model, y, x,
|
|
pf("%s_PADOUT%i", prefix, i),
|
|
pf("%s_DIFFI_IN%i", prefix, 1-i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
if (num_devs > 1) {
|
|
for (i = 0; i <= 1; i++) {
|
|
rc = add_switch(model, y, x,
|
|
pf("%s_PADOUT%i", prefix, i+2),
|
|
pf("%s_DIFFI_IN%i", prefix, 3-i),
|
|
0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_io(struct fpga_model *model)
|
|
{
|
|
int x, y, rc;
|
|
|
|
RC_CHECK(model);
|
|
for (x = 0; x < model->x_width; x++) {
|
|
if (has_device(model, /*y*/ 0, x, DEV_IOB)) {
|
|
rc = init_io_tile(model, 0, x);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
if (has_device(model, model->y_height - BOT_OUTER_ROW, x,
|
|
DEV_IOB)) {
|
|
rc = init_io_tile(model,
|
|
model->y_height-BOT_OUTER_ROW, x);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
for (y = 0; y < model->y_height; y++) {
|
|
if (has_device(model, y, /*x*/ 0, DEV_IOB)) {
|
|
rc = init_io_tile(model, y, 0);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
if (has_device(model, y, model->x_width - RIGHT_OUTER_O,
|
|
DEV_IOB)) {
|
|
rc = init_io_tile(model,
|
|
y, model->x_width - RIGHT_OUTER_O);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
const char* wire_base(enum wire_type w)
|
|
{
|
|
switch (w) {
|
|
case W_NL1: return "NL1";
|
|
case W_NR1: return "NR1";
|
|
case W_EL1: return "EL1";
|
|
case W_ER1: return "ER1";
|
|
case W_SL1: return "SL1";
|
|
case W_SR1: return "SR1";
|
|
case W_WL1: return "WL1";
|
|
case W_WR1: return "WR1";
|
|
|
|
case W_NN2: return "NN2";
|
|
case W_NE2: return "NE2";
|
|
case W_EE2: return "EE2";
|
|
case W_SE2: return "SE2";
|
|
case W_SS2: return "SS2";
|
|
case W_SW2: return "SW2";
|
|
case W_WW2: return "WW2";
|
|
case W_NW2: return "NW2";
|
|
|
|
case W_NN4: return "NN4";
|
|
case W_NE4: return "NE4";
|
|
case W_EE4: return "EE4";
|
|
case W_SE4: return "SE4";
|
|
case W_SS4: return "SS4";
|
|
case W_SW4: return "SW4";
|
|
case W_WW4: return "WW4";
|
|
case W_NW4: return "NW4";
|
|
}
|
|
HERE();
|
|
return "";
|
|
}
|
|
|
|
enum wire_type base2wire(const char* str)
|
|
{
|
|
if (!strncmp(str, "NL1", 3)) return W_NL1;
|
|
if (!strncmp(str, "NR1", 3)) return W_NR1;
|
|
if (!strncmp(str, "EL1", 3)) return W_EL1;
|
|
if (!strncmp(str, "ER1", 3)) return W_ER1;
|
|
if (!strncmp(str, "SL1", 3)) return W_SL1;
|
|
if (!strncmp(str, "SR1", 3)) return W_SR1;
|
|
if (!strncmp(str, "WL1", 3)) return W_WL1;
|
|
if (!strncmp(str, "WR1", 3)) return W_WR1;
|
|
|
|
if (!strncmp(str, "NN2", 3)) return W_NN2;
|
|
if (!strncmp(str, "NE2", 3)) return W_NE2;
|
|
if (!strncmp(str, "EE2", 3)) return W_EE2;
|
|
if (!strncmp(str, "SE2", 3)) return W_SE2;
|
|
if (!strncmp(str, "SS2", 3)) return W_SS2;
|
|
if (!strncmp(str, "SW2", 3)) return W_SW2;
|
|
if (!strncmp(str, "WW2", 3)) return W_WW2;
|
|
if (!strncmp(str, "NW2", 3)) return W_NW2;
|
|
|
|
if (!strncmp(str, "NN4", 3)) return W_NN4;
|
|
if (!strncmp(str, "NE4", 3)) return W_NE4;
|
|
if (!strncmp(str, "EE4", 3)) return W_EE4;
|
|
if (!strncmp(str, "SE4", 3)) return W_SE4;
|
|
if (!strncmp(str, "SS4", 3)) return W_SS4;
|
|
if (!strncmp(str, "SW4", 3)) return W_SW4;
|
|
if (!strncmp(str, "WW4", 3)) return W_WW4;
|
|
if (!strncmp(str, "NW4", 3)) return W_NW4;
|
|
|
|
HERE();
|
|
return 0;
|
|
}
|
|
|
|
static int rotate_num(int cur, int off, int first, int last)
|
|
{
|
|
if (cur+off > last)
|
|
return first + (cur+off-last-1) % ((last+1)-first);
|
|
if (cur+off < first)
|
|
return last - (first-(cur+off)-1) % ((last+1)-first);
|
|
return cur+off;
|
|
}
|
|
|
|
enum wire_type rotate_wire(enum wire_type cur, int off)
|
|
{
|
|
if (W_IS_LEN1(cur))
|
|
return rotate_num(cur, off, FIRST_LEN1, LAST_LEN1);
|
|
if (W_IS_LEN2(cur))
|
|
return rotate_num(cur, off, FIRST_LEN2, LAST_LEN2);
|
|
if (W_IS_LEN4(cur))
|
|
return rotate_num(cur, off, FIRST_LEN4, LAST_LEN4);
|
|
EXIT(1);
|
|
}
|
|
|
|
enum wire_type wire_to_len(enum wire_type w, int first_len)
|
|
{
|
|
if (W_IS_LEN1(w))
|
|
return w-FIRST_LEN1 + first_len;
|
|
if (W_IS_LEN2(w))
|
|
return w-FIRST_LEN2 + first_len;
|
|
if (W_IS_LEN4(w))
|
|
return w-FIRST_LEN4 + first_len;
|
|
EXIT(1);
|
|
}
|
|
|
|
static const char* routing_wirestr(enum extra_wires wire,
|
|
int routing_io, int gclk_brk)
|
|
{
|
|
if (routing_io) {
|
|
if (wire == GFAN0) return "INT_IOI_GFAN0";
|
|
if (wire == GFAN1) return "INT_IOI_GFAN1";
|
|
if (wire == LW + LI_A5) return "INT_IOI_LOGICIN_B4";
|
|
if (wire == LW + LI_B4) return "INT_IOI_LOGICIN_B10";
|
|
}
|
|
if (gclk_brk) {
|
|
switch (wire) {
|
|
case GCLK0: return "GCLK0_BRK";
|
|
case GCLK1: return "GCLK1_BRK";
|
|
case GCLK2: return "GCLK2_BRK";
|
|
case GCLK3: return "GCLK3_BRK";
|
|
case GCLK4: return "GCLK4_BRK";
|
|
case GCLK5: return "GCLK5_BRK";
|
|
case GCLK6: return "GCLK6_BRK";
|
|
case GCLK7: return "GCLK7_BRK";
|
|
case GCLK8: return "GCLK8_BRK";
|
|
case GCLK9: return "GCLK9_BRK";
|
|
case GCLK10: return "GCLK10_BRK";
|
|
case GCLK11: return "GCLK11_BRK";
|
|
case GCLK12: return "GCLK12_BRK";
|
|
case GCLK13: return "GCLK13_BRK";
|
|
case GCLK14: return "GCLK14_BRK";
|
|
case GCLK15: return "GCLK15_BRK";
|
|
default: ;
|
|
}
|
|
}
|
|
return fpga_wire2str(wire);
|
|
}
|
|
|
|
static int init_routing_tile(struct fpga_model *model, int y, int x)
|
|
{
|
|
int i, routing_io, gclk_brk, from_wire, to_wire, is_bidir, rc;
|
|
struct fpga_tile* tile;
|
|
|
|
RC_CHECK(model);
|
|
tile = YX_TILE(model, y, x);
|
|
routing_io = (tile->type == IO_ROUTING || tile->type == ROUTING_IO_L);
|
|
gclk_brk = (tile->type == ROUTING_BRK || tile->type == BRAM_ROUTING_BRK);
|
|
|
|
// KEEP1
|
|
for (i = X_A1; i <= M_WE; i++) {
|
|
rc = add_switch(model, y, x, "KEEP1_WIRE",
|
|
logicin_s(i, routing_io), 0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
rc = add_switch(model, y, x, "KEEP1_WIRE", "FAN_B", 0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
for (i = 0; i <= 1; i++) {
|
|
rc = add_switch(model, y, x, "KEEP1_WIRE",
|
|
pf("CLK%i", i), 0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, y, x, "KEEP1_WIRE",
|
|
pf("SR%i", i), 0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, y, x,
|
|
"KEEP1_WIRE", routing_wirestr(GFAN0+i, routing_io, gclk_brk), 0 /* bidir */);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
|
|
for (i = 0; i < model->num_bitpos; i++) {
|
|
from_wire = model->sw_bitpos[i].from;
|
|
to_wire = model->sw_bitpos[i].to;
|
|
is_bidir = model->sw_bitpos[i].bidir;
|
|
if (routing_io) {
|
|
if (from_wire == GFAN0 || from_wire == GFAN1) {
|
|
from_wire = VCC_WIRE;
|
|
is_bidir = 0;
|
|
} else if (to_wire == GFAN0 || to_wire == GFAN1)
|
|
is_bidir = 0;
|
|
}
|
|
rc = add_switch(model, y, x,
|
|
routing_wirestr(from_wire, routing_io, gclk_brk),
|
|
routing_wirestr(to_wire, routing_io, gclk_brk),
|
|
is_bidir);
|
|
if (rc) RC_FAIL(model, rc);
|
|
if (is_bidir) {
|
|
rc = add_switch(model, y, x,
|
|
routing_wirestr(to_wire, routing_io, gclk_brk),
|
|
routing_wirestr(from_wire, routing_io, gclk_brk),
|
|
/* bidir */ 1);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
if (routing_io) {
|
|
// These switches don't come out of the general model because
|
|
// they are bidir there and skipped on the reverse side, but
|
|
// fall back to regular unidir switches in the io tiles. Can
|
|
// be cleaned up one day.
|
|
rc = add_switch(model, y, x, "LOGICIN_B6", "INT_IOI_GFAN0", 0);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, y, x, "LOGICIN_B35", "INT_IOI_GFAN0", 0);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, y, x, "LOGICIN_B51", "INT_IOI_GFAN1", 0);
|
|
if (rc) RC_FAIL(model, rc);
|
|
rc = add_switch(model, y, x, "LOGICIN_B53", "INT_IOI_GFAN1", 0);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
{ const int logicin_b[] = {20, 21, 28, 36, 44, 52, 60, 62};
|
|
for (i = 0; i < sizeof(logicin_b)/sizeof(*logicin_b); i++) {
|
|
rc = add_switch(model, y, x,
|
|
pf("LOGICIN_B%i", logicin_b[i]),
|
|
pf("LOGICIN%i", logicin_b[i]),
|
|
/* bidir */ 0);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_routing(struct fpga_model *model)
|
|
{
|
|
int x, y, rc;
|
|
|
|
RC_CHECK(model);
|
|
for (x = 0; x < model->x_width; x++) {
|
|
if (!is_atx(X_ROUTING_COL, model, x))
|
|
continue;
|
|
for (y = TOP_IO_TILES; y < model->y_height-BOT_IO_TILES; y++) {
|
|
if (is_aty(Y_ROW_HORIZ_AXSYMM|Y_CHIP_HORIZ_REGS,
|
|
model, y))
|
|
continue;
|
|
rc = init_routing_tile(model, y, x);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
int replicate_routing_switches(struct fpga_model *model)
|
|
{
|
|
struct fpga_tile* tile;
|
|
int x, y, first_y, first_x, rc;
|
|
|
|
RC_CHECK(model);
|
|
first_y = -1;
|
|
for (x = 0; x < model->x_width; x++) {
|
|
if (!is_atx(X_ROUTING_COL, model, x))
|
|
continue;
|
|
for (y = TOP_IO_TILES; y < model->y_height-BOT_IO_TILES; y++) {
|
|
if (is_aty(Y_ROW_HORIZ_AXSYMM|Y_CHIP_HORIZ_REGS,
|
|
model, y))
|
|
continue;
|
|
tile = YX_TILE(model, y, x);
|
|
// Some tiles are different so we cannot include
|
|
// them in the high-speed replication scheme.
|
|
if (tile->type == IO_ROUTING || tile->type == ROUTING_IO_L
|
|
|| tile->type == ROUTING_BRK || tile->type == BRAM_ROUTING_BRK) {
|
|
rc = init_routing_tile(model, y, x);
|
|
if (rc) RC_FAIL(model, rc);
|
|
continue;
|
|
}
|
|
if (first_y == -1) {
|
|
first_y = y;
|
|
first_x = x;
|
|
rc = init_routing_tile(model, y, x);
|
|
if (rc) RC_FAIL(model, rc);
|
|
continue;
|
|
}
|
|
rc = replicate_switches_and_names(model,
|
|
first_y, first_x, y, x);
|
|
if (rc) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_center(struct fpga_model *model)
|
|
{
|
|
int i, j, rc;
|
|
|
|
RC_CHECK(model);
|
|
{ const char* pairs[] =
|
|
{ "CLKC_CKLR%i", "CLKC_GCLK%i",
|
|
"CLKC_CKTB%i", "CLKC_GCLK%i",
|
|
"CLKC_PLL_L%i", "CLKC_GCLK%i",
|
|
"CLKC_PLL_U%i", "CLKC_GCLK%i",
|
|
"CLKC_SEL%i_PLL", "S_GCLK_SITE%i",
|
|
"I0_GCLK_SITE%i", "O_GCLK_SITE%i",
|
|
"O_GCLK_SITE%i", "CLKC_GCLK_MAIN%i" };
|
|
int i_dest[2][16] =
|
|
{{ 0,1,2,4,3,5,6,7,8,9,10,12,11,13,14,15 },
|
|
{ 1,0,3,5,2,4,7,6,9,8,11,13,10,12,15,14 }};
|
|
|
|
for (i = 0; i < sizeof(pairs)/sizeof(*pairs)/2; i++) {
|
|
for (j = 0; j <= 15; j++) {
|
|
if ((rc = add_switch(model, model->center_y, model->center_x,
|
|
pf(pairs[i*2], j), pf(pairs[i*2+1], j),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
for (j = 0; j <= 15; j++) {
|
|
if ((rc = add_switch(model, model->center_y, model->center_x,
|
|
pf("CLKC_GCLK%i", j), pf("I0_GCLK_SITE%i", i_dest[0][j]),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, model->center_y, model->center_x,
|
|
pf("CLKC_GCLK%i", j), pf("I1_GCLK_SITE%i", i_dest[1][j]),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
|
|
{ const char *to[] = {
|
|
"CLK_PLL_LOCK_LT0", "CLK_PLL_LOCK_LT1",
|
|
"CLK_PLL_LOCK_RT0", "CLK_PLL_LOCK_RT1" };
|
|
const char *from[] = {
|
|
"PLL_LOCK_BOT0", "PLL_LOCK_BOT1",
|
|
"PLL_LOCK_TOP0", "PLL_LOCK_TOP1" };
|
|
for (i = 0; i < sizeof(to)/sizeof(*to); i++) {
|
|
for (j = 0; j < sizeof(from)/sizeof(*from); j++) {
|
|
if ((rc = add_switch(model, model->center_y,
|
|
model->center_x-CENTER_CMTPLL_O,
|
|
from[j], to[i], /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}}
|
|
|
|
{ const char* pairs[] = {
|
|
"PLL_LOCK_BOT0", "PLL_LOCK_TOP2",
|
|
"PLL_LOCK_BOT1", "PLL_LOCK_TOP2",
|
|
"PLL_LOCK_TOP0", "PLL_LOCK_BOT2",
|
|
"PLL_LOCK_TOP1", "PLL_LOCK_BOT2" };
|
|
|
|
for (i = 0; i < sizeof(pairs)/sizeof(*pairs)/2; i++) {
|
|
if ((rc = add_switch(model, model->center_y,
|
|
model->center_x-CENTER_CMTPLL_O,
|
|
pairs[i*2], pairs[i*2+1],
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
|
|
{ const char *to[] = {
|
|
"REGC_CLKPLL_IO_LT0", "REGC_CLKPLL_IO_LT1",
|
|
"REGC_CLKPLL_IO_RT0", "REGC_CLKPLL_IO_RT1",
|
|
"REGC_PLLCLK_UP_OUT0", "REGC_PLLCLK_UP_OUT1" };
|
|
for (i = 0; i <= 3; i++) {
|
|
for (j = 0; j < sizeof(to)/sizeof(*to); j++) {
|
|
if ((rc = add_switch(model, model->center_y,
|
|
model->center_x-CENTER_CMTPLL_O,
|
|
pf("REGC_PLLCLK_DN_IN%i", i), to[j],
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}}
|
|
{ const char *to[] = {
|
|
"REGC_CLKPLL_IO_LT0", "REGC_CLKPLL_IO_LT1",
|
|
"REGC_CLKPLL_IO_RT0", "REGC_CLKPLL_IO_RT1",
|
|
"REGC_PLLCLK_DN_OUT0", "REGC_PLLCLK_DN_OUT1" };
|
|
for (i = 0; i <= 3; i++) {
|
|
for (j = 0; j < sizeof(to)/sizeof(*to); j++) {
|
|
if ((rc = add_switch(model, model->center_y,
|
|
model->center_x-CENTER_CMTPLL_O,
|
|
pf("REGC_PLLCLK_UP_IN%i", i), to[j],
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_hclk(struct fpga_model *model)
|
|
{
|
|
int x, y, i, rc;
|
|
|
|
RC_CHECK(model);
|
|
for (x = 0; x < model->x_width; x++) {
|
|
if (!is_atx(X_ROUTING_COL, model, x))
|
|
continue;
|
|
for (y = TOP_IO_TILES; y < model->y_height-BOT_IO_TILES; y++) {
|
|
if (!is_aty(Y_ROW_HORIZ_AXSYMM, model, y))
|
|
continue;
|
|
for (i = 0; i <= 15; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("HCLK_GCLK%i_INT", i), pf("HCLK_GCLK%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("HCLK_GCLK%i_INT", i), pf("HCLK_GCLK_UP%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_logicout_fw(struct fpga_model *model)
|
|
{
|
|
int i, x, y, rc;
|
|
|
|
RC_CHECK(model);
|
|
for (x = 0; x < model->x_width; x++) {
|
|
if (is_atx(X_FABRIC_BRAM_VIA_COL|X_FABRIC_MACC_VIA_COL, model, x)) {
|
|
for (y = TOP_IO_TILES; y < model->y_height - BOT_IO_TILES; y++) {
|
|
if (is_aty(Y_CHIP_HORIZ_REGS|Y_ROW_HORIZ_AXSYMM, model, y))
|
|
continue;
|
|
for (i = 0; i <= 23; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("INT_INTERFACE_LOGICOUT_%i", i), pf("INT_INTERFACE_LOGICOUT%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
if (is_atx(X_CENTER_ROUTING_COL, model, x)) {
|
|
for (y = TOP_IO_TILES; y < model->y_height - BOT_IO_TILES; y++) {
|
|
if (!is_aty(Y_ROW_HORIZ_AXSYMM, model, y))
|
|
continue;
|
|
for (i = 0; i <= 23; i++) {
|
|
if ((rc = add_switch(model, y-1, model->center_x-CENTER_LOGIC_O,
|
|
pf("INT_INTERFACE_LOGICOUT_%i", i), pf("INT_INTERFACE_LOGICOUT%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y+1, model->center_x-CENTER_LOGIC_O,
|
|
pf("INT_INTERFACE_LOGICOUT_%i", i), pf("INT_INTERFACE_LOGICOUT%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
if (is_atx(X_LEFT_IO_DEVS_COL, model, x)) {
|
|
for (y = TOP_IO_TILES; y < model->y_height - BOT_IO_TILES; y++) {
|
|
if (is_aty(Y_CHIP_HORIZ_REGS|Y_ROW_HORIZ_AXSYMM, model, y)
|
|
|| has_device(model, y, LEFT_IO_DEVS, DEV_ILOGIC))
|
|
continue;
|
|
if (has_device(model, y, LEFT_IO_DEVS, DEV_OCT_CALIBRATE)) {
|
|
for (i = 0; i <= 23; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("INT_INTERFACE_LOCAL_LOGICOUT_%i", i), pf("INT_INTERFACE_LOCAL_LOGICOUT%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
continue;
|
|
}
|
|
// todo: this is probably not right...
|
|
if (y == model->center_y-1 || y == model->center_y-2
|
|
|| y < TOP_IO_TILES + HALF_ROW
|
|
|| y == model->y_height-BOT_INNER_IO)
|
|
continue;
|
|
for (i = 0; i <= 23; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("INT_INTERFACE_LOGICOUT_%i", i), pf("INT_INTERFACE_LOGICOUT%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
if (is_atx(X_RIGHT_IO_DEVS_COL, model, x)) {
|
|
for (y = TOP_IO_TILES; y < model->y_height - BOT_IO_TILES; y++) {
|
|
if (is_aty(Y_CHIP_HORIZ_REGS|Y_ROW_HORIZ_AXSYMM, model, y)
|
|
|| has_device(model, y, model->x_width-RIGHT_IO_DEVS_O, DEV_ILOGIC))
|
|
continue;
|
|
if (has_device(model, y, model->x_width-RIGHT_IO_DEVS_O, DEV_BSCAN)
|
|
|| has_device(model, y, model->x_width-RIGHT_IO_DEVS_O, DEV_ICAP)
|
|
|| has_device(model, y, model->x_width-RIGHT_IO_DEVS_O, DEV_SLAVE_SPI)) {
|
|
for (i = 0; i <= 23; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("INT_INTERFACE_LOCAL_LOGICOUT_%i", i), pf("INT_INTERFACE_LOCAL_LOGICOUT%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
continue;
|
|
}
|
|
// todo: this is probably not right...
|
|
if (y == model->center_y-1 || y == model->center_y-2
|
|
|| y < TOP_IO_TILES + HALF_ROW)
|
|
continue;
|
|
for (i = 0; i <= 23; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("INT_INTERFACE_LOGICOUT_%i", i), pf("INT_INTERFACE_LOGICOUT%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_bram(struct fpga_model *model)
|
|
{
|
|
int i, x, y, tile0_to_3, wire_num, rc;
|
|
|
|
RC_CHECK(model);
|
|
for (x = 0; x < model->x_width; x++) {
|
|
if (!is_atx(X_FABRIC_BRAM_COL, model, x))
|
|
continue;
|
|
for (y = TOP_IO_TILES; y < model->y_height - BOT_IO_TILES; y++) {
|
|
if (!has_device(model, y, x, DEV_BRAM16))
|
|
continue;
|
|
{ const char* pairs[] = {
|
|
"BRAM_CLK%c_INT1", "RAMB16BWER_CLK%c",
|
|
"BRAM_CLK%c_INT1", "RAMB8BWER_0_CLK%c",
|
|
"BRAM_CLK%c_INT2", "RAMB8BWER_1_CLK%c" };
|
|
for (i = 0; i < sizeof(pairs)/sizeof(*pairs)/2; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf(pairs[i*2], '0'+0), pf(pairs[i*2+1], 'A'+0),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf(pairs[i*2], '0'+1), pf(pairs[i*2+1], 'A'+1),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
{ const char *s[] = {
|
|
"BRAM_SR0_INT1", "RAMB16BWER_RSTA",
|
|
"BRAM_SR0_INT1", "RAMB8BWER_0_RSTA",
|
|
"BRAM_SR0_INT2", "RAMB8BWER_1_RSTA",
|
|
"BRAM_SR1_INT1", "RAMB16BWER_RSTB",
|
|
"BRAM_SR1_INT1", "RAMB8BWER_0_RSTB",
|
|
"BRAM_SR1_INT2", "RAMB8BWER_1_RSTB" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y, x, s[i*2],
|
|
s[i*2+1], /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
for (i = BI_FIRST; i <= BI_LAST; i++) {
|
|
fdev_bram_inbit(BW+i, &tile0_to_3, &wire_num);
|
|
if (tile0_to_3 == -1) { HERE(); continue; }
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("BRAM_LOGICINB%i_INT%i", wire_num, tile0_to_3),
|
|
fpga_wire2str(BW+i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if (fdev_is_bram8_inwire(i)) {
|
|
fdev_bram_inbit(BW+(B8_0|i), &tile0_to_3, &wire_num);
|
|
if (tile0_to_3 == -1) { HERE(); continue; }
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("BRAM_LOGICINB%i_INT%i", wire_num, tile0_to_3),
|
|
fpga_wire2str(BW+(B8_0|i)),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
|
|
fdev_bram_inbit(BW+(B8_1|i), &tile0_to_3, &wire_num);
|
|
if (tile0_to_3 == -1) { HERE(); continue; }
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("BRAM_LOGICINB%i_INT%i", wire_num, tile0_to_3),
|
|
fpga_wire2str(BW+(B8_1|i)),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
for (i = BO_FIRST; i <= BO_LAST; i++) {
|
|
fdev_bram_outbit(BW+i, &tile0_to_3, &wire_num);
|
|
if (tile0_to_3 == -1) { HERE(); continue; }
|
|
if ((rc = add_switch(model, y, x,
|
|
fpga_wire2str(BW+i),
|
|
pf("BRAM_LOGICOUT%i_INT%i", wire_num, tile0_to_3),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if (fdev_is_bram8_outwire(i)) {
|
|
fdev_bram_outbit(BW+(B8_0|i), &tile0_to_3, &wire_num);
|
|
if (tile0_to_3 == -1) { HERE(); continue; }
|
|
if ((rc = add_switch(model, y, x,
|
|
fpga_wire2str(BW+(B8_0|i)),
|
|
pf("BRAM_LOGICOUT%i_INT%i", wire_num, tile0_to_3),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
|
|
fdev_bram_outbit(BW+(B8_1|i), &tile0_to_3, &wire_num);
|
|
if (tile0_to_3 == -1) { HERE(); continue; }
|
|
if ((rc = add_switch(model, y, x,
|
|
fpga_wire2str(BW+(B8_1|i)),
|
|
pf("BRAM_LOGICOUT%i_INT%i", wire_num, tile0_to_3),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_macc(struct fpga_model *model)
|
|
{
|
|
int i, x, y, tile0_to_3, wire_num, rc;
|
|
|
|
RC_CHECK(model);
|
|
for (x = 0; x < model->x_width; x++) {
|
|
if (!is_atx(X_FABRIC_MACC_COL, model, x))
|
|
continue;
|
|
for (y = TOP_IO_TILES; y < model->y_height - BOT_IO_TILES; y++) {
|
|
if (!has_device(model, y, x, DEV_MACC))
|
|
continue;
|
|
{ const char *s[] = {
|
|
"MACC_SR0_INT0", "RSTD_DSP48A1_SITE",
|
|
"MACC_SR0_INT1", "RSTP_DSP48A1_SITE",
|
|
"MACC_SR0_INT2", "RSTC_DSP48A1_SITE",
|
|
"MACC_SR0_INT3", "RSTA_DSP48A1_SITE",
|
|
"MACC_SR1_INT0", "RSTCARRYIN_DSP48A1_SITE",
|
|
"MACC_SR1_INT1", "RSTOPMODE_DSP48A1_SITE",
|
|
"MACC_SR1_INT2", "RSTM_DSP48A1_SITE",
|
|
"MACC_SR1_INT3", "RSTB_DSP48A1_SITE",
|
|
"MACC_CLK0_INT2", "CLK_DSP48A1_SITE" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y, x, s[i*2],
|
|
s[i*2+1], /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
for (i = MI_FIRST; i <= MI_LAST; i++) {
|
|
fdev_macc_inbit(MW+i, &tile0_to_3, &wire_num);
|
|
if (tile0_to_3 == -1) { HERE(); continue; }
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("MACC_LOGICINB%i_INT%i", wire_num, tile0_to_3),
|
|
fpga_wire2str(MW+i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
for (i = MO_FIRST; i <= MO_LAST; i++) {
|
|
fdev_macc_outbit(MW+i, &tile0_to_3, &wire_num);
|
|
if (tile0_to_3 == -1) { HERE(); continue; }
|
|
if ((rc = add_switch(model, y, x,
|
|
fpga_wire2str(MW+i),
|
|
pf("MACC_LOGICOUT%i_INT%i", wire_num, tile0_to_3),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
if (y != TOP_IO_TILES+3) { // exclude topmost macc dev
|
|
if ((rc = add_switch(model, y, x,
|
|
"CARRYOUT_DSP48A1_SITE", "CARRYOUT_DSP48A1_B_SITE",
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
for (i = 0; i <= 17; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("BCOUT%i_DSP48A1_SITE", i), pf("BCOUT%i_DSP48A1_B_SITE", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
for (i = 0; i <= 47; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("PCOUT%i_DSP48A1_SITE", i), pf("PCOUT%i_DSP48A1_B_SITE", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_topbot_tterm_gclk(struct fpga_model *model)
|
|
{
|
|
int i, rc;
|
|
|
|
RC_CHECK(model);
|
|
for (i = 0; i <= 15; i++) {
|
|
if ((rc = add_switch(model, TOP_INNER_ROW, model->center_x + CENTER_X_PLUS_1, pf("IOI_TTERM_GCLK%i", i),
|
|
"BUFPLL_TOP_GCLK0", /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, TOP_INNER_ROW, model->center_x + CENTER_X_PLUS_1, pf("IOI_TTERM_GCLK%i", i),
|
|
"BUFPLL_TOP_GCLK1", /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, model->y_height - BOT_INNER_ROW, model->center_x + CENTER_X_PLUS_1, pf("IOI_BTERM_GCLK%i", i),
|
|
"BUFPLL_BOT_GCLK0", /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, model->y_height - BOT_INNER_ROW, model->center_x + CENTER_X_PLUS_1, pf("IOI_BTERM_GCLK%i", i),
|
|
"BUFPLL_BOT_GCLK1", /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_bufio_tile(struct fpga_model *model, int y, int x, const char *s1, const char *s2)
|
|
{
|
|
int i, j, rc;
|
|
|
|
RC_CHECK(model);
|
|
{ static const char *s[] = {
|
|
"I_BUFIO2_%s_SITE%i", "O_BUFIO2_%s_SITE%i",
|
|
"I_BUFIO2_%s_SITE%i", "ODIV_BUFIO2_%s_SITE%i",
|
|
"IFB_BUFIO2FB_%s_SITE%i", "OFB_BUFIO2FB_%s_SITE%i" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
for (j = 0; j <= 7; j++) {
|
|
if ((rc = add_switch(model, y, x, pf(s[i*2], s2, j),
|
|
pf(s[i*2+1], s2, j), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}}
|
|
{ static const char *s[] = {
|
|
"O_BUFIO2_%s_SITE%i", "%s_IOCLKOUT%i",
|
|
"ODIV_BUFIO2_%s_SITE%i", "%s_CKPIN_OUT%i",
|
|
"OE_BUFIO2_%s_SITE%i", "%s_IOCEOUT%i",
|
|
"OFB_BUFIO2FB_%s_SITE%i", "%s_CLK_FEEDBACK%i" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
for (j = 0; j <= 7; j++) {
|
|
if ((rc = add_switch(model, y, x, pf(s[i*2], s2, j),
|
|
pf(s[i*2+1], s1, j), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}}
|
|
{ static const char *s[] = {
|
|
"%s_CFB%i", "IFB_BUFIO2FB_%s_SITE%i",
|
|
"%s_CFB1_%i", "IB_BUFIO2FB_%s_SITE%i",
|
|
"%s_DFB%i", "IFB_BUFIO2FB_%s_SITE%i",
|
|
"%s_DFB%i", "I_BUFIO2_%s_SITE%i",
|
|
"%s_GTPFB%i", "IFB_BUFIO2FB_%s_SITE%i",
|
|
"%s_CLKPIN%i", "IB_BUFIO2_%s_SITE%i",
|
|
"%s_CLKPIN%i", "I_BUFIO2_%s_SITE%i" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
for (j = 0; j <= 7; j++) {
|
|
if ((rc = add_switch(model, y, x, pf(s[i*2], s1, j),
|
|
pf(s[i*2+1], s2, j), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}}
|
|
for (j = 0; j <= 7; j++) {
|
|
if ((rc = add_switch(model, y, x, pf("%s_VCC", s1),
|
|
pf("IB_BUFIO2_%s_SITE%i", s2, j),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
{ int n[] = { 1, 0, 3, 2, 5, 4, 7, 6 };
|
|
const char *s[] = {
|
|
"%s_CLKPIN%i", "IFB_BUFIO2FB_%s_SITE%i",
|
|
"%s_DFB%i", "IB_BUFIO2_%s_SITE%i" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
for (j = 0; j <= 7; j++) {
|
|
if ((rc = add_switch(model, y, x, pf(s[i*2], s1, j),
|
|
pf(s[i*2+1], s2, n[j]),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}}
|
|
{ int n1[] = { 1, 0, 3, 2, 0, 0, 2, 2};
|
|
int n2[] = { 4, 4, 6, 6, 1, 1, 3, 3};
|
|
int n3[] = { 5, 5, 7, 7, 5, 4, 7, 6};
|
|
const char *s[] = {
|
|
"%s_CLKPIN%i", "IB_BUFIO2_%s_SITE%i",
|
|
"%s_CLKPIN%i", "I_BUFIO2_%s_SITE%i" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
for (j = 0; j <= 7; j++) {
|
|
if ((rc = add_switch(model, y, x, pf(s[i*2], s1, j),
|
|
pf(s[i*2+1], s2, n1[j]),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x, pf(s[i*2], s1, j),
|
|
pf(s[i*2+1], s2, n2[j]),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x, pf(s[i*2], s1, j),
|
|
pf(s[i*2+1], s2, n3[j]),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_bufio(struct fpga_model *model)
|
|
{
|
|
int y, x, j, rc;
|
|
const char *s1, *s2;
|
|
const int lr_n[] = { 0, 0, 2, 2, 4, 5, 6, 7 };
|
|
|
|
RC_CHECK(model);
|
|
|
|
y = TOP_OUTER_ROW;
|
|
x = model->center_x-CENTER_CMTPLL_O;
|
|
s1 = "REGT";
|
|
s2 = "TOP";
|
|
rc = init_bufio_tile(model, y, x, s1, s2);
|
|
if (rc) RC_FAIL(model, rc);
|
|
for (j = 0; j <= 7; j++) {
|
|
if ((rc = add_switch(model, y, x, pf("%s_GTPCLK%i", s1, j),
|
|
pf("I_BUFIO2_%s_SITE%i", s2, j), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
|
|
y = model->center_y;
|
|
x = LEFT_OUTER_COL;
|
|
s1 = "REGL";
|
|
s2 = "LEFT";
|
|
rc = init_bufio_tile(model, y, x, s1, s2);
|
|
if (rc) RC_FAIL(model, rc);
|
|
for (j = 0; j <= 7; j++) {
|
|
if ((rc = add_switch(model, y, x, pf("%s_GTPCLK%i", s1, lr_n[j]),
|
|
pf("I_BUFIO2_%s_SITE%i", s2, j), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
|
|
y = model->center_y;
|
|
x = model->x_width-RIGHT_OUTER_O;
|
|
s1 = "REGR";
|
|
s2 = "RIGHT";
|
|
rc = init_bufio_tile(model, y, x, s1, s2);
|
|
if (rc) RC_FAIL(model, rc);
|
|
for (j = 0; j <= 7; j++) {
|
|
if ((rc = add_switch(model, y, x, pf("%s_GTPCLK%i", s1, lr_n[j]),
|
|
pf("I_BUFIO2_%s_SITE%i", s2, j), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
|
|
y = model->y_height-BOT_OUTER_ROW;
|
|
x = model->center_x-CENTER_CMTPLL_O;
|
|
s1 = "REGB";
|
|
s2 = "BOT";
|
|
rc = init_bufio_tile(model, y, x, s1, s2);
|
|
if (rc) RC_FAIL(model, rc);
|
|
for (j = 0; j <= 7; j++) {
|
|
if ((rc = add_switch(model, y, x, pf("%s_GTPCLK%i", s1, j),
|
|
pf("I_BUFIO2_%s_SITE%i", s2, j), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_bscan(struct fpga_model *model)
|
|
{
|
|
int y, x, num_bscan_devs, i, j, rc;
|
|
const char *s[] = {
|
|
"BSCAN%i_CAPTURE_PINWIRE",
|
|
"BSCAN%i_DRCK_PINWIRE",
|
|
"BSCAN%i_RESET_PINWIRE",
|
|
"BSCAN%i_RUNTEST_PINWIRE",
|
|
"BSCAN%i_SHIFT_PINWIRE",
|
|
"BSCAN%i_TCK_PINWIRE",
|
|
"BSCAN%i_TDI_PINWIRE",
|
|
"BSCAN%i_TMS_PINWIRE",
|
|
"BSCAN%i_UPDATE_PINWIRE",
|
|
"BSCAN%i_SEL_PINWIRE" };
|
|
|
|
RC_CHECK(model);
|
|
x = model->x_width-RIGHT_IO_DEVS_O;
|
|
for (y = TOP_IO_TILES; y < model->y_height-BOT_IO_TILES; y++) {
|
|
num_bscan_devs = has_device(model, y, x, DEV_BSCAN);
|
|
if (!num_bscan_devs) continue;
|
|
if (num_bscan_devs != 2) {
|
|
HERE();
|
|
continue;
|
|
}
|
|
for (i = 0; i <= 1; i++) {
|
|
for (j = 0; j < sizeof(s)/sizeof(*s); j++) {
|
|
if ((rc = add_switch(model, y, x, pf(s[j], i+1),
|
|
pf("INT_INTERFACE_LOCAL_LOGICOUT_%i", i*10+j), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
if ((rc = add_switch(model, y, x, pf("INT_INTERFACE_LOCAL_LOGICBIN%i", i+1),
|
|
pf("BSCAN%i_TDO_PINWIRE", i+1), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_dcm(struct fpga_model *model)
|
|
{
|
|
int y, x, i, j, k, num_devs, rc;
|
|
|
|
RC_CHECK(model);
|
|
x = model->center_x-CENTER_CMTPLL_O;
|
|
for (y = TOP_IO_TILES; y < model->y_height-BOT_IO_TILES; y++) {
|
|
num_devs = has_device(model, y, x, DEV_DCM);
|
|
if (!num_devs) continue;
|
|
if (num_devs != 2) {
|
|
HERE();
|
|
continue;
|
|
}
|
|
for (i = 0; i <= 1; i++) { // 2 dcm devs
|
|
for (j = 0; j <= 9; j++) { // 10 clk outs
|
|
for (k = 0; k <= 15; k++) { // 16 hclk dests
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("DCM%i_CLKOUT%i", i+1, j),
|
|
pf("DCM_HCLK%i", k),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
}
|
|
for (k = 0; k <= 15; k++) { // 16 hclk dests
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("DCM_FABRIC_CLK%i", k), pf("DCM_HCLK%i", k),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("DCM_HCLK%i", k), pf("DCM_HCLK%i_N", k),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if (y > model->center_y) { // lower half
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("DCM_HCLK%i_N", k), pf("PLL_CLK_CASC_TOP%i", k),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("PLL_CLK_CASC_BOT%i", k), pf("PLL_CLK_CASC_TOP%i", k),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
} else { // upper half
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("DCM_HCLK%i_N", k), pf("PLL_CLK_CASC_BOT%i", k),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("PLL_CLK_CASC_TOP%i", k), pf("PLL_CLK_CASC_BOT%i", k),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
for (i = 0; i <= 7; i++) { // 8 wires
|
|
for (j = 1; j <= 2; j++) { // dcm 1 and 2
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("DCM_CLK_FEEDBACK_LR_TOP%i", i), pf("DCM%i_CLKFB", j),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("DCM_CLK_FEEDBACK_TB_BOT%i", i), pf("DCM%i_CLKFB", j),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("DCM_CLK_INDIRECT_LR_TOP%i", i), pf("DCM%i_CLKIN", j),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("DCM_CLK_INDIRECT_TB_BOT%i", i), pf("DCM%i_CLKIN", j),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
// logicin
|
|
{ const char *clb1_in0_to_62[] = {
|
|
/* 0*/ 0, 0, 0, 0, 0, "DCM2_STSADRS3", 0, 0,
|
|
/* 8*/ 0, 0, 0, 0, "DCM2_STSADRS2", 0, 0, "DCM2_PSINCDEC",
|
|
/*16*/ "DCM2_STSADRS0", 0, 0, 0, 0, 0, 0, 0,
|
|
/*24*/ "DCM2_PSEN", "DCM2_SE_CLK_IN1", 0, 0, 0, 0, 0, 0,
|
|
/*32*/ 0, 0, "DCM2_CTLOSC2", 0, "DCM2_STSADRS4", 0, 0, 0,
|
|
/*40*/ 0, 0, "DCM2_FREEZEDFS", 0, 0, 0, 0, "DCM2_RST",
|
|
/*48*/ 0, 0, 0, 0, 0, 0, "DCM2_CTLOSC1", 0,
|
|
/*56*/ 0, "DCM2_SE_CLK_IN0", 0, 0, 0, 0, "DCM2_STSADRS1" };
|
|
const char *clb2_in0_to_62[] = {
|
|
/* 0*/ 0, 0, "DCM_FABRIC_CLK13", "DCM_FABRIC_CLK5", 0, "DCM1_STSADRS3", 0, 0,
|
|
/* 8*/ "DCM_FABRIC_CLK15", 0, 0, 0, "DCM1_STSADRS2", 0, 0, "DCM1_PSINCDEC",
|
|
/*16*/ "DCM1_STSADRS0", 0, 0, "DCM_FABRIC_CLK7", 0, 0, 0, 0,
|
|
/*24*/ "DCM1_PSEN", "DCM1_SE_CLK_IN1", "DCM_FABRIC_CLK2", 0, "DCM_FABRIC_CLK8", "DCM_FABRIC_CLK9", "DCM_FABRIC_CLK3", 0,
|
|
/*32*/ "DCM_FABRIC_CLK10", 0, "DCM1_CTLOSC2", 0, "DCM1_STSADRS4", 0, 0, "DCM_FABRIC_CLK14",
|
|
/*40*/ 0, "DCM_FABRIC_CLK4", "DCM1_FREEZEDFS", 0, "DCM_FABRIC_CLK0", 0, 0, "DCM1_RST",
|
|
/*48*/ 0, 0, 0, 0, "DCM_FABRIC_CLK12", 0, "DCM1_CTLOSC1", "DCM_FABRIC_CLK1",
|
|
/*56*/ 0, "DCM1_SE_CLK_IN0", "DCM_FABRIC_CLK6", "DCM_FABRIC_CLK11", 0, 0, "DCM1_STSADRS1" };
|
|
for (i = 0; i < 63; i++) {
|
|
if (clb1_in0_to_62[i])
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("DCM_CLB1_LOGICINB%i", i), clb1_in0_to_62[i],
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if (clb2_in0_to_62[i])
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("DCM_CLB2_LOGICINB%i", i), clb2_in0_to_62[i],
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
// logicout
|
|
for (i = 0; i <= 7; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("DCM1_STATUS%i", i), pf("DCM_CLB2_LOGICOUT%i", 16+i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("DCM2_STATUS%i", i), pf("DCM_CLB1_LOGICOUT%i", 16+i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
if ((rc = add_switch(model, y, x, "DCM1_LOCKED",
|
|
"DCM_CLB2_LOGICOUT14", /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x, "DCM1_PSDONE",
|
|
"DCM_CLB2_LOGICOUT15", /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x, "DCM2_LOCKED",
|
|
"DCM_CLB1_LOGICOUT14", /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x, "DCM2_PSDONE",
|
|
"DCM_CLB1_LOGICOUT15", /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
|
|
{ const char *s[] = {
|
|
"CLK0", "CLK90", "CLK180", "CLK270", "CLK2X",
|
|
"CLK2X180", "CLKDV", "CLKFX", "CLKFX180", "CONCUR" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s); i++) {
|
|
for (j = 1; j <= 2; j++) { // dcm1 and dcm2
|
|
if ((rc = add_switch(model, y, x, pf("DCM%i_%s", j, s[i]),
|
|
pf("DCM%i_CLKOUT%i", j, i), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x, pf("DCM%i_%s", j, s[i]),
|
|
pf("DCM%i_CLK_TO_PLL", j), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x, pf("DCM%i_%s", j, s[i]),
|
|
pf("DCM%i_%s_TEST", j, s[i]), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x, pf("DCM%i_%s_TEST", j, s[i]),
|
|
pf("DCM_%i_TESTCLK_PINWIRE", j==1?1:0), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}}
|
|
|
|
{ const char *s[] = {
|
|
"CMT_DCM_LOCK_UP0", "CMT_DCM_LOCK_DN0",
|
|
"CMT_DCM_LOCK_UP1", "CMT_DCM_LOCK_DN1",
|
|
"DCM_IOCLK_UP0", "DCM_IOCLK_DOWN0",
|
|
"DCM_IOCLK_UP1", "DCM_IOCLK_DOWN1",
|
|
"DCM_IOCLK_UP2", "DCM_IOCLK_DOWN2",
|
|
"DCM_IOCLK_UP3", "DCM_IOCLK_DOWN3",
|
|
"DCM1_CLK_TO_PLL", "DCM_1_MUXED_CLK_PINWIRE",
|
|
"DCM2_CLK_TO_PLL", "DCM_0_MUXED_CLK_PINWIRE",
|
|
"DCM_CLB1_CLK0", "DCM2_CLK_FROM_BUFG0",
|
|
"DCM_CLB1_CLK1", "DCM2_CLK_FROM_BUFG1",
|
|
"DCM_CLB2_CLK0", "DCM1_CLK_FROM_BUFG0",
|
|
"DCM_CLB2_CLK1", "DCM1_CLK_FROM_BUFG1",
|
|
"DCM1_GFAN1", "DCM2_PSCLK",
|
|
"DCM2_GFAN1", "DCM1_PSCLK" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y, x, s[i*2], s[i*2+1],
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
{ const char *s[] = {
|
|
"DCM%i_CLK_FROM_BUFG0", "DCM%i_CLKFB",
|
|
"DCM%i_CLK_FROM_BUFG1", "DCM%i_CLKIN",
|
|
"DCM%i_CLK_FROM_PLL", "DCM%i_CLKIN",
|
|
"DCM%i_SE_CLK_IN0", "DCM%i_CLKFB",
|
|
"DCM%i_SE_CLK_IN1", "DCM%i_CLKIN" };
|
|
for (j = 1; j <= 2; j++) { // dcm1 and dcm2
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y, x, pf(s[i*2], j), pf(s[i*2+1], j),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}}
|
|
if (y > model->center_y) { // lower half
|
|
const char *s[] = {
|
|
"CMT_DCM_LOCK_UP2", "CMT_DCM_LOCK_DN2",
|
|
"DCM_IOCLK_UP4", "DCM_IOCLK_DOWN4",
|
|
"DCM_IOCLK_UP5", "DCM_IOCLK_DOWN5" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y, x, s[i*2], s[i*2+1],
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
} else { // upper half
|
|
const char *s[] = {
|
|
"CMT_DCM_LOCK_DN2", "CMT_DCM_LOCK_UP2",
|
|
"DCM_IOCLK_DOWN4", "DCM_IOCLK_UP4",
|
|
"DCM_IOCLK_DOWN5", "DCM_IOCLK_UP5" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y, x, s[i*2], s[i*2+1],
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_pll(struct fpga_model *model)
|
|
{
|
|
int y, x, i, j, num_devs, rc;
|
|
|
|
RC_CHECK(model);
|
|
x = model->center_x-CENTER_CMTPLL_O;
|
|
for (y = TOP_IO_TILES; y < model->y_height-BOT_IO_TILES; y++) {
|
|
num_devs = has_device(model, y, x, DEV_PLL);
|
|
if (!num_devs) continue;
|
|
if (num_devs != 1) {
|
|
HERE();
|
|
continue;
|
|
}
|
|
for (i = 0; i <= 5; i++) { // 6 clk outs
|
|
for (j = 0; j <= 15; j++) { // 16 hclk dests
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CMT_PLL_CLKOUT%i", i),
|
|
pf("CMT_PLL_HCLK%i", j),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
for (i = 0; i <= 15; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
"CMT_CLKFB",
|
|
pf("CMT_PLL_HCLK%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
"CMT_SE_CLK_OUT",
|
|
pf("CMT_PLL_HCLK%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CMT_FABRIC_CLK%i", i),
|
|
pf("CMT_PLL_HCLK%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CMT_PLL_HCLK%i", i),
|
|
pf("CMT_PLL_HCLK%i_E", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if (y < model->center_y) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CMT_PLL_HCLK%i_E", i),
|
|
pf("CLK_PLLCASC_OUT%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("PLL_CLK_CASC_IN%i", i),
|
|
pf("CLK_PLLCASC_OUT%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
} else {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CMT_PLL_HCLK%i_E", i),
|
|
pf("PLL_CLK_CASC_IN%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
for (i = 0; i <= 7; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CMT_PLL_CLK_FEEDBACK_LRBOT%i", i),
|
|
"CMT_CLKMUX_CLKFB",
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("PLL_CLK_FEEDBACK_TB%i", i),
|
|
"CMT_CLKMUX_CLKFB",
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
for (i = 0; i <= 7; i++) {
|
|
const char *to = i < 4 ? "CMT_CLKMUX_CLKREF" : "CMT_CLKMUX_CLKIN2";
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CMT_PLL_CLK_INDIRECT_LRBOT%i", i), to,
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("PLL_CLK_INDIRECT_TB%i", i), to,
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
for (i = 0; i <= 5; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CMT_PLL_CLKOUTDCM%i", i),
|
|
"CMT_CLK_TO_DCM1",
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CMT_PLL_CLKOUTDCM%i", i),
|
|
"CMT_CLK_TO_DCM2",
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
{ const char *s[] = {
|
|
"CMT_CLKFB", "CMT_CLKMUX_CLKFB",
|
|
"CMT_CLK_FROM_BUFG0", "CMT_CLKMUX_CLKREF",
|
|
"CMT_CLK_FROM_BUFG1", "CMT_CLKMUX_CLKIN2",
|
|
"CMT_CLK_FROM_BUFG2", "CMT_CLKMUX_CLKFB",
|
|
"CMT_CLK_FROM_DCM1", "CMT_CLKMUX_CLKIN2",
|
|
"CMT_CLK_FROM_DCM2", "CMT_CLKMUX_CLKIN2",
|
|
"CMT_CLKMUX_CLKFB", "CMT_CLKMUX_CLKFB_TEST",
|
|
"CMT_CLKMUX_CLKFB", "CMT_PLL_CLKFBIN",
|
|
"CMT_CLKMUX_CLKIN2", "CMT_PLL_CLKIN2",
|
|
"CMT_CLKMUX_CLKREF", "CMT_CLKMUX_CLKREF_TEST",
|
|
"CMT_CLKMUX_CLKREF", "CMT_PLL_CLKIN1",
|
|
"CMT_CLK_TO_DCM1", "CMT_PLL_SKEWCLKIN1",
|
|
"CMT_CLK_TO_DCM2", "CMT_PLL_SKEWCLKIN2",
|
|
"CMT_PLL_CLKFB", "CMT_CLKFB",
|
|
"CMT_PLL_CLKFBDCM", "CMT_CLKMUX_CLKFB",
|
|
"CMT_PLL_CLKFBDCM", "CMT_PLL_CLKFBDCM_TEST",
|
|
"CMT_PLL_CLKOUT0", "CMT_CLKMUX_CLKFB",
|
|
"CMT_PLL_CLKOUT0", "PLLCASC_CLKOUT0",
|
|
"CMT_PLL_CLKOUT1", "PLLCASC_CLKOUT1",
|
|
"CMT_PLL_LOCKED", "CMT_PLL_LOCK_DN1",
|
|
"CMT_PLL_LOCKED", "CMT_PLL_LOCK_UP1",
|
|
"CMT_SE_CLKIN0", "CMT_CLKMUX_CLKIN2",
|
|
"CMT_SE_CLKIN1", "CMT_CLKMUX_CLKFB",
|
|
"CMT_TEST_CLK", "CMT_SE_CLK_OUT",
|
|
"PLLCASC_CLKOUT0", "PLL_IOCLK_DN2",
|
|
"PLLCASC_CLKOUT0", "PLL_IOCLK_UP2",
|
|
"PLLCASC_CLKOUT1", "PLL_IOCLK_DN3",
|
|
"PLLCASC_CLKOUT1", "PLL_IOCLK_UP3",
|
|
"PLL_LOCKED", "CMT_PLL_LOCKED",
|
|
"PLL_LOCKED", "PLL_CLB1_LOGICOUT18",
|
|
"PLL_VCC", "PLL_REL",
|
|
"PLL_CLB1_CLK0", "CMT_CLK_FROM_BUFG0",
|
|
"PLL_CLB1_CLK1", "CMT_CLK_FROM_BUFG1",
|
|
"PLL_CLB2_CLK0", "CMT_CLK_FROM_BUFG2",
|
|
"PLL_CLB2_GFAN1", "PLL_DCLK" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
s[i*2], s[i*2+1], /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
// logicin
|
|
{ const char *clb1_in0_to_62[] = {
|
|
/* 0*/ 0, 0, "13", "6", "1", 0, 0, 0,
|
|
/* 8*/ "15", 0, 0, 0, 0, 0, 0, 0,
|
|
/*16*/ 0, 0, 0, "7", 0, 0, 0, 0,
|
|
/*24*/ 0, 0, "3", 0, "8", "9", "4", 0,
|
|
/*32*/ "10", 0, 0, 0, 0, 0, 0, "14",
|
|
/*40*/ 0, "5", 0, 0, "0", 0, 0, 0,
|
|
/*48*/ 0, 0, 0, 0, "12", 0, 0, "2",
|
|
/*56*/ 0, 0, 0, "11", 0, 0, 0 };
|
|
const char *clb2_in0_to_62[] = {
|
|
/* 0*/ 0, "PLL_DI14", 0, 0, 0, "PLL_DADDR3", 0, "PLL_DADDR1",
|
|
/* 8*/ 0, 0, 0, 0, "PLL_DI0", 0, 0, "PLL_DADDR0",
|
|
/*16*/ 0, "PLL_DI4", 0, "PLL_DI12", 0, 0, 0, 0,
|
|
/*24*/ "PLL_DEN", "PLL_DI2", "PLL_DI8", "PLL_DI13", "CMT_SE_CLKIN0", 0, "PLL_DI10", 0,
|
|
/*32*/ "PLL_DI15", 0, "PLL_DI1", 0, "PLL_DADDR4", "PLL_CLKINSEL", "PLL_DI3", 0,
|
|
/*40*/ 0, "PLL_DI11", "PLL_DADDR2", 0, "PLL_DI5", 0, 0, 0,
|
|
/*48*/ "PLL_DI9", 0, 0, 0, "PLL_RST", 0, 0, "PLL_DI6",
|
|
/*56*/ 0, "PLL_DI7", "PLL_DWE", "CMT_SE_CLKIN1", 0, 0, 0 };
|
|
for (i = 0; i < 63; i++) {
|
|
if (clb1_in0_to_62[i])
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("PLL_CLB1_LOGICINB%i", i),
|
|
pf("CMT_FABRIC_CLK%s", clb1_in0_to_62[i]),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if (clb2_in0_to_62[i])
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("PLL_CLB2_LOGICINB%i", i), clb2_in0_to_62[i],
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
// logicout
|
|
for (i = 0; i <= 15; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("PLL_DO%i", i),
|
|
pf("PLL_CLB1_LOGICOUT%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
if ((rc = add_switch(model, y, x, "PLL_DRDY",
|
|
"PLL_CLB1_LOGICOUT16", /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_center_hclk(struct fpga_model *model)
|
|
{
|
|
int y, x, i, rc;
|
|
|
|
RC_CHECK(model);
|
|
x = model->center_x;
|
|
for (y = TOP_IO_TILES; y < model->y_height-BOT_IO_TILES; y++) {
|
|
if (!is_aty(Y_ROW_HORIZ_AXSYMM, model, y))
|
|
continue;
|
|
for (i = 0; i <= 15; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CLKV_GCLKH_L%i", i),
|
|
pf("I_BUFH_LEFT_SITE%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CLKV_GCLKH_R%i", i),
|
|
pf("I_BUFH_RIGHT_SITE%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CLKV_GCLKH_MAIN%i_FOLD", i),
|
|
pf("CLKV_GCLKH_L%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CLKV_GCLKH_MAIN%i_FOLD", i),
|
|
pf("CLKV_GCLKH_R%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("I_BUFH_LEFT_SITE%i", i),
|
|
pf("O_BUFH_LEFT_SITE%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("I_BUFH_RIGHT_SITE%i", i),
|
|
pf("O_BUFH_RIGHT_SITE%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("O_BUFH_LEFT_SITE%i", i),
|
|
pf("CLKV_BUFH_LEFT_L%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("O_BUFH_RIGHT_SITE%i", i),
|
|
pf("CLKV_BUFH_RIGHT_R%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("REGV_PLL_HCLK%i", i),
|
|
pf("CLKV_GCLKH_L%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("REGV_PLL_HCLK%i", i),
|
|
pf("CLKV_GCLKH_R%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_center_midbuf(struct fpga_model *model)
|
|
{
|
|
int y, x, i, rc;
|
|
|
|
RC_CHECK(model);
|
|
x = model->center_x;
|
|
for (y = TOP_IO_TILES; y < model->y_height-BOT_IO_TILES; y++) {
|
|
if (!is_atyx(YX_CENTER_MIDBUF, model, y, x))
|
|
continue;
|
|
if (y < model->center_y) {
|
|
for (i = 0; i <= 7; i++) {
|
|
if ((rc = add_switch(model, y-1, x,
|
|
pf("CLKV_CKPIN_BUF%i", i),
|
|
pf("CLKV_MIDBUF_TOP_CKPIN%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
} else {
|
|
for (i = 0; i <= 7; i++) {
|
|
if ((rc = add_switch(model, y+1, x,
|
|
pf("CLKV_MIDBUF_BOT_CKPIN%i", i),
|
|
pf("CLKV_CKPIN_BOT_BUF%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
for (i = 0; i <= 15; i++) {
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CLKV_GCLK_MAIN%i", i),
|
|
pf("CLKV_MIDBUF_GCLK%i", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x,
|
|
pf("CLKV_MIDBUF_GCLK%i", i),
|
|
pf("CLKV_GCLK_MAIN%i_BUF", i),
|
|
/*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_center_reg_tile(struct fpga_model *model, int y, int x, const char *s1, const char *s2)
|
|
{
|
|
int i, j, rc;
|
|
|
|
RC_CHECK(model);
|
|
{ static const char *s[] = {
|
|
"%s_CKPIN_OUT%i", "%s_CKPIN%i",
|
|
"%s_CKPIN_OUT%i", "%s_CLK_INDIRECT%i",
|
|
"%s_GND", "%s_IOCLKOUT%i",
|
|
"%s_VCC", "%s_CKPIN%i",
|
|
"%s_VCC", "%s_CLK_FEEDBACK%i",
|
|
"%s_VCC", "%s_CLK_INDIRECT%i" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
for (j = 0; j <= 7; j++) {
|
|
if ((rc = add_switch(model, y, x, pf(s[i*2], s1, j),
|
|
pf(s[i*2+1], s1, j), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_center_reg_tblr(struct fpga_model *model)
|
|
{
|
|
int y, x, i, j, rc;
|
|
const char *s1, *s2;
|
|
|
|
RC_CHECK(model);
|
|
|
|
//
|
|
// top
|
|
//
|
|
|
|
y = TOP_OUTER_ROW;
|
|
x = model->center_x-CENTER_CMTPLL_O;
|
|
s1 = "REGT";
|
|
s2 = "TOP";
|
|
rc = init_center_reg_tile(model, y, x, s1, s2);
|
|
if (rc) RC_FAIL(model, rc);
|
|
for (i = 0; i <= 5; i++) {
|
|
for (j = 0; j <= 1; j++) {
|
|
if ((rc = add_switch(model, y, x, pf("REGT_PLL_IOCLK_UP%i", i),
|
|
pf("PLLIN_BUFPLL%i_TOP_SITE", j), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if (i < 3) {
|
|
if ((rc = add_switch(model, y, x, pf("REGT_LOCKIN%i", i),
|
|
pf("LOCKED_BUFPLL%i_TOP_SITE", j), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
}
|
|
{ const char *s[] = {
|
|
"REGT_VCC", "REGT_PLLCLK0",
|
|
"REGT_VCC", "REGT_PLLCLK1",
|
|
"IOCLK_BUFPLL0_TOP_SITE", "REGT_PLLCLK0",
|
|
"IOCLK_BUFPLL1_TOP_SITE", "REGT_PLLCLK1",
|
|
"LOCK_BUFPLL0_TOP_SITE", "REGT_LOCK0",
|
|
"LOCK_BUFPLL1_TOP_SITE", "REGT_LOCK1",
|
|
"REGT_GCLK0", "GCLK_BUFPLL0_TOP_SITE",
|
|
"REGT_GCLK1", "GCLK_BUFPLL1_TOP_SITE",
|
|
"SERDESSTROBE_BUFPLL0_TOP_SITE", "REGT_CEOUT0",
|
|
"SERDESSTROBE_BUFPLL1_TOP_SITE", "REGT_CEOUT1" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y, x, pf(s[i*2]),
|
|
pf(s[i*2+1]), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
// y+1
|
|
for (i = 0; i <= 7; i++) {
|
|
if ((rc = add_switch(model, y+1, x, pf("REGT_TTERM_CLKPIN%i", i),
|
|
pf("REGT_TTERM_CKPIN%i", i), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
{ const char *s[] = {
|
|
"REGT_TTERM_PLL_CEOUT0_N", "REGT_TTERM_PLL_CEOUT0",
|
|
"REGT_TTERM_PLL_CEOUT1_N", "REGT_TTERM_PLL_CEOUT1",
|
|
"REGT_TTERM_PLL_CLKOUT0_N", "REGT_TTERM_PLL_CLKOUT0",
|
|
"REGT_TTERM_PLL_CLKOUT1_N", "REGT_TTERM_PLL_CLKOUT1" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y+1, x, pf(s[i*2]),
|
|
pf(s[i*2+1]), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
|
|
//
|
|
// left
|
|
//
|
|
|
|
y = model->center_y;
|
|
x = LEFT_OUTER_COL;
|
|
s1 = "REGL";
|
|
s2 = "LEFT";
|
|
rc = init_center_reg_tile(model, y, x, s1, s2);
|
|
if (rc) RC_FAIL(model, rc);
|
|
{ const char *s[] = {
|
|
"REGL_VCC", "REGL_PLL_CLKOUT0_LEFT",
|
|
"REGL_VCC", "REGL_PLL_CLKOUT1_LEFT",
|
|
"IOCLK_BUFPLL0_LEFT_SITE", "REGL_PLL_CLKOUT0_LEFT",
|
|
"IOCLK_BUFPLL1_LEFT_SITE", "REGL_PLL_CLKOUT1_LEFT",
|
|
"LOCK_BUFPLL0_LEFT_SITE", "REGL_LOCK0",
|
|
"LOCK_BUFPLL1_LEFT_SITE", "REGL_LOCK1",
|
|
"REGL_CLKPLL0", "PLLIN_BUFPLL0_LEFT_SITE",
|
|
"REGL_CLKPLL1", "PLLIN_BUFPLL1_LEFT_SITE",
|
|
"REGL_GCLK0", "GCLK_BUFPLL0_LEFT_SITE",
|
|
"REGL_GCLK1", "GCLK_BUFPLL1_LEFT_SITE",
|
|
"REGL_GCLK2", "PLLIN_BUFPLL0_LEFT_SITE",
|
|
"REGL_GCLK3", "PLLIN_BUFPLL1_LEFT_SITE",
|
|
"REGL_LOCKED0", "LOCKED_BUFPLL0_LEFT_SITE",
|
|
"REGL_LOCKED1", "LOCKED_BUFPLL1_LEFT_SITE",
|
|
"REGL_PCI_IRDY_IOB", "REGL_PCI_IRDY_PINW",
|
|
"REGL_PCI_TRDY_IOB", "REGL_PCI_TRDY_PINW",
|
|
"SERDESSTROBE_BUFPLL0_LEFT_SITE", "REGL_PLL_CEOUT0_LEFT",
|
|
"SERDESSTROBE_BUFPLL1_LEFT_SITE", "REGL_PLL_CEOUT1_LEFT" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y, x, pf(s[i*2]),
|
|
pf(s[i*2+1]), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
// x+1
|
|
for (i = 0; i <= 7; i++) {
|
|
if ((rc = add_switch(model, y, x+1, pf("REGH_LTERM_CLKPIN%i", i),
|
|
pf("REGH_LTERM_CKPIN%i", i), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
{ const char *s[] = {
|
|
"REGH_LTERM_PLL_CEOUT0_W", "REGH_LTERM_PLL_CEOUT0",
|
|
"REGH_LTERM_PLL_CEOUT1_W", "REGH_LTERM_PLL_CEOUT1",
|
|
"REGH_LTERM_PLL_CLKOUT0_W", "REGH_LTERM_PLL_CLKOUT0",
|
|
"REGH_LTERM_PLL_CLKOUT1_W", "REGH_LTERM_PLL_CLKOUT1" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y, x+1, pf(s[i*2]),
|
|
pf(s[i*2+1]), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
|
|
//
|
|
// right
|
|
//
|
|
|
|
y = model->center_y;
|
|
x = model->x_width-RIGHT_OUTER_O;
|
|
s1 = "REGR";
|
|
s2 = "RIGHT";
|
|
rc = init_center_reg_tile(model, y, x, s1, s2);
|
|
if (rc) RC_FAIL(model, rc);
|
|
{ const char *s[] = {
|
|
"REGR_VCC", "REGR_PLLCLK0",
|
|
"REGR_VCC", "REGR_PLLCLK1",
|
|
"IOCLK_BUFPLL0_RIGHT_SITE", "REGR_PLLCLK0",
|
|
"IOCLK_BUFPLL1_RIGHT_SITE", "REGR_PLLCLK1",
|
|
"LOCK_BUFPLL0_RIGHT_SITE", "REGR_LOCK0",
|
|
"LOCK_BUFPLL1_RIGHT_SITE", "REGR_LOCK1",
|
|
"REGR_CLKPLL0", "PLLIN_BUFPLL0_RIGHT_SITE",
|
|
"REGR_CLKPLL1", "PLLIN_BUFPLL1_RIGHT_SITE",
|
|
"REGR_GCLK0", "GCLK_BUFPLL0_RIGHT_SITE",
|
|
"REGR_GCLK1", "GCLK_BUFPLL1_RIGHT_SITE",
|
|
"REGR_GCLK2", "PLLIN_BUFPLL0_RIGHT_SITE",
|
|
"REGR_GCLK3", "PLLIN_BUFPLL1_RIGHT_SITE",
|
|
"REGR_LOCKED0", "LOCKED_BUFPLL0_RIGHT_SITE",
|
|
"REGR_LOCKED1", "LOCKED_BUFPLL1_RIGHT_SITE",
|
|
"REGR_PCI_IRDY_IOB", "REGR_PCI_IRDY_PINW",
|
|
"REGR_PCI_TRDY_IOB", "REGR_PCI_TRDY_PINW",
|
|
"SERDESSTROBE_BUFPLL0_RIGHT_SITE", "REGR_CEOUT0",
|
|
"SERDESSTROBE_BUFPLL1_RIGHT_SITE", "REGR_CEOUT1" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y, x, pf(s[i*2]),
|
|
pf(s[i*2+1]), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
// x-1
|
|
for (i = 0; i <= 7; i++) {
|
|
if ((rc = add_switch(model, y, x-1, pf("REGH_RTERM_CLKPIN%i", i),
|
|
pf("REGH_RTERM_CKPIN%i", i), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
{ const char *s[] = {
|
|
"REGH_RTERM_PLL_CEOUT0_E", "REGH_RTERM_PLL_CEOUT0",
|
|
"REGH_RTERM_PLL_CEOUT1_E", "REGH_RTERM_PLL_CEOUT1",
|
|
"REGH_RTERM_PLL_CLKOUT0_E", "REGH_RTERM_PLL_CLKOUT0",
|
|
"REGH_RTERM_PLL_CLKOUT1_E", "REGH_RTERM_PLL_CLKOUT1" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y, x-1, pf(s[i*2]),
|
|
pf(s[i*2+1]), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
|
|
//
|
|
// bottom
|
|
//
|
|
|
|
y = model->y_height-BOT_OUTER_ROW;
|
|
x = model->center_x-CENTER_CMTPLL_O;
|
|
s1 = "REGB";
|
|
s2 = "BOT";
|
|
rc = init_center_reg_tile(model, y, x, s1, s2);
|
|
if (rc) RC_FAIL(model, rc);
|
|
for (i = 0; i <= 5; i++) {
|
|
for (j = 0; j <= 1; j++) {
|
|
if ((rc = add_switch(model, y, x, pf("REGB_PLL_IOCLK_DOWN%i", i),
|
|
pf("PLLIN_BUFPLL%i_BOT_SITE", j), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if (i < 3) {
|
|
if ((rc = add_switch(model, y, x, pf("REGB_LOCKIN%i", i),
|
|
pf("LOCKED_BUFPLL%i_BOT_SITE", j), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
}
|
|
{ const char *s[] = {
|
|
"REGB_VCC", "REGB_PLLCLK0",
|
|
"REGB_VCC", "REGB_PLLCLK1",
|
|
"IOCLK_BUFPLL0_BOT_SITE", "REGB_PLLCLK0",
|
|
"IOCLK_BUFPLL1_BOT_SITE", "REGB_PLLCLK1",
|
|
"LOCK_BUFPLL0_BOT_SITE", "REGB_LOCK0",
|
|
"LOCK_BUFPLL1_BOT_SITE", "REGB_LOCK1",
|
|
"REGB_GCLK0", "GCLK_BUFPLL0_BOT_SITE",
|
|
"REGB_GCLK1", "GCLK_BUFPLL1_BOT_SITE",
|
|
"SERDESSTROBE_BUFPLL0_BOT_SITE", "REGB_CEOUT0",
|
|
"SERDESSTROBE_BUFPLL1_BOT_SITE", "REGB_CEOUT1" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y, x, pf(s[i*2]),
|
|
pf(s[i*2+1]), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
// y-1
|
|
for (i = 0; i <= 7; i++) {
|
|
if ((rc = add_switch(model, y-1, x, pf("REGB_BTERM_CLKPIN%i", i),
|
|
pf("REGB_BTERM_CKPIN%i", i), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
{ const char *s[] = {
|
|
"REGB_BTERM_PLL_CEOUT0_S", "REGB_BTERM_PLL_CEOUT0",
|
|
"REGB_BTERM_PLL_CEOUT1_S", "REGB_BTERM_PLL_CEOUT1",
|
|
"REGB_BTERM_PLL_CLKOUT0_S", "REGB_BTERM_PLL_CLKOUT0",
|
|
"REGB_BTERM_PLL_CLKOUT1_S", "REGB_BTERM_PLL_CLKOUT1" };
|
|
for (i = 0; i < sizeof(s)/sizeof(*s)/2; i++) {
|
|
if ((rc = add_switch(model, y-1, x, pf(s[i*2]),
|
|
pf(s[i*2+1]), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}}
|
|
RC_RETURN(model);
|
|
}
|
|
|
|
static int init_center_topbot_cfb_dfb(struct fpga_model *model)
|
|
{
|
|
const int x_enum[] = { model->center_x-CENTER_LOGIC_O,
|
|
model->center_x+CENTER_X_PLUS_2 };
|
|
int y, i, x_i, rc;
|
|
|
|
RC_CHECK(model);
|
|
y = TOP_INNER_ROW;
|
|
for (x_i = 0; x_i < sizeof(x_enum)/sizeof(*x_enum); x_i++) {
|
|
for (i = 0; i <= 1; i++) {
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("IOI_REGT_CFB1_M%i_S", i+1),
|
|
pf("IOI_REGT_CFB1_M%i", i+1), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("IOI_REGT_CFB1_S%i_S", i+1),
|
|
pf("IOI_REGT_CFB1_S%i", i+1), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("IOI_REGT_CFB_M%i_S", i+1),
|
|
pf("IOI_REGT_CFB_M%i", i+1), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("IOI_REGT_CFB_S%i_S", i+1),
|
|
pf("IOI_REGT_CFB_S%i", i+1), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("IOI_REGT_DFB_M%i_S", i+1),
|
|
pf("IOI_REGT_DFB_M%i", i+1), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("IOI_REGT_DFB_S%i_S", i+1),
|
|
pf("IOI_REGT_DFB_S%i", i+1), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("IOI_REGT_DQSN%i_S", i),
|
|
pf("IOI_REGT_DQSN%i", i), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("IOI_REGT_DQSP%i_S", i),
|
|
pf("IOI_REGT_DQSP%i", i), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("TTERM_IOIBOT_IBUF%i", i),
|
|
pf("IOI_REGT_CLKPIN%i", i+2), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("TTERM_IOIUP_IBUF%i", i),
|
|
pf("IOI_REGT_CLKPIN%i", i), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
|
|
y = model->y_height-BOT_INNER_ROW;
|
|
for (x_i = 0; x_i < sizeof(x_enum)/sizeof(*x_enum); x_i++) {
|
|
for (i = 0; i <= 3; i++) {
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("BTERM_CLB_CFB1_%i_N", i+4),
|
|
pf("BTERM_CLB_CFB1_%i", i+4), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("BTERM_CLB_CFB%i_N", i+4),
|
|
pf("BTERM_CLB_CFB%i", i+4), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("BTERM_CLB_DFB%i_N", i+4),
|
|
pf("BTERM_CLB_DFB%i", i+4), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
for (i = 0; i <= 1; i++) {
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("BTERM_CLB_DQSN%i_N", i+2),
|
|
pf("BTERM_CLB_DQSN%i", i+2), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("BTERM_CLB_DQSP%i_N", i+2),
|
|
pf("BTERM_CLB_DQSP%i", i+2), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("BTERM_IOIBOT_IBUF%i", i),
|
|
pf("BTERM_CLB_CLKPIN%i", i+4), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
if ((rc = add_switch(model, y, x_enum[x_i], pf("BTERM_IOIUP_IBUF%i", i),
|
|
pf("BTERM_CLB_CLKPIN%i", i+6), /*bidir*/ 0))) RC_FAIL(model, rc);
|
|
}
|
|
}
|
|
RC_RETURN(model);
|
|
}
|