blob: f663078c3625803b07a860562f4641f329c12b39 [file] [log] [blame]
// Copyright lowRISC contributors.
// Licensed under the Apache License, Version 2.0, see LICENSE for details.
// SPDX-License-Identifier: Apache-2.0
{ name: "otbn"
clock_primary: "clk_i"
bus_device: "tlul"
bus_host: "none"
param_list: [
{ name: "RegFile",
type: "otbn_pkg::regfile_e",
default: "otbn_pkg::RegFileFF",
desc: "Selection of the register file implementation. See otbn_pkg.sv."
local: "false",
expose: "true"
}
]
interrupt_list: [
{ name: "done"
desc: "OTBN has completed the operation"
}
]
alert_list: [
{ name: "fatal"
desc: "A fatal error. Fatal alerts are non-recoverable and will be asserted until a hard reset."
}
{ name: "recov"
desc: "A recoverable error. Just sent once (as the processor stops)."
}
]
inter_signal_list: [
// TODO: CSRNG isn't committed yet, depends on assumed interface done for
// AES in https://github.com/lowRISC/opentitan/pull/2140.
/*{ name: "entropy",
type: "req_rsp",
act: "req",
package: "csrng_pkg",
struct: "csrng_entropy",
width: "1"
},*/
// OTBN is not performing any operation and can be clock/power-gated.
{ name: "idle",
type: "uni",
struct: "logic",
width: "1",
act: "req",
},
],
regwidth: "32"
registers: [
{ name: "CMD"
desc: "command register",
swaccess: "r0w1c",
hwaccess: "hro",
hwext: "true",
hwqe: "true",
fields: [
{ bits: "0"
name: "start"
desc: '''
Start the operation
The completion is signalled by the done interrupt.
'''
tags: [
// Don't write this field in the automated CSR tests. Doing so will
// start OTBN, but we won't have initialised its memory with any
// code, so we'll get Xs on its interfaces and everything will be
// a bit of a mess!
"excl:CsrAllTests:CsrExclWrite"
]
}
{ bits: "1",
name: "dummy",
desc: '''
Reggen doesn't generate sub-fields with only a single field
specified; instead, the whole register is taken as a field, leading
to signals like `hw2reg.status.d` instead of
`hw2reg.status.start.d`. Since we expect to add more commands later,
we force the generation of fields with this dummy field for now.
'''
}
],
}
{ name: "STATUS",
desc: "Status",
swaccess: "ro",
hwaccess: "hwo",
hwext: "true",
fields: [
{ bits: "0",
name: "busy",
desc: "OTBN is performing an operation."
}
{ bits: "1",
name: "dummy",
desc: '''
See !!CMD.dummy for details.
'''
}
]
} // register : status
{ name: "ERR_BITS",
desc: '''
Error bitfield. Reads as non-zero if an error was seen during OTBN
operation
''',
swaccess: "ro",
hwaccess: "hwo",
fields: [
{ bits: "0",
name: "bad_data_addr"
desc: '''
A DMEM read or write occurred with an out of bounds or unaligned
address.
'''
}
{ bits: "1",
name: "bad_insn_addr"
desc: '''
An IMEM read or write occurred with an out of bounds or unaligned
address.
'''
}
{ bits: "2",
name: "call_stack"
desc: '''
A instruction tried to pop from an empty call stack or push to a
full call stack.
'''
}
{ bits: "3",
name: "illegal_insn"
desc: '''
One of the following happened:
<ul>
<li>An instruction being excuted had an invalid encoding.</li>
<li>An access occurred for an invalid CSR or WSR.</li>
<li>
A CSR or WSR access occurred that is not permitted (e.g. writing
to a read-only CSR or WSR).
</li>
</ul>
'''
}
{ bits: "4",
name: "loop"
desc: '''
One of the following happened:
<ul>
<li>A loop was started with an iteration count of zero.</li>
<li>
The final instruction of a loop was a branch or another loop.
</li>
<li>
A new loop tried to push to a full loop stack (loop nesting
level too deep).
</li>
</ul>
'''
}
{ bits: "5",
name: "fatal_imem"
desc: "A fatal failure was seen on an instruction fetch."
}
{ bits: "6",
name: "fatal_dmem"
desc: "A fatal failure was seen on a DMEM read."
}
{ bits: "7",
name: "fatal_reg"
desc: "A fatal failure was seen on a GPR or WDR read."
}
]
} // register : err_bits
{ name: "START_ADDR",
desc: "Start byte address in the instruction memory",
swaccess: "wo",
hwaccess: "hro",
fields: [
{ bits: "31:0",
name: "start_addr",
resval: 0,
desc: '''
Byte address in the instruction memory OTBN starts to execute from
when instructed to do so with the !!CMD.start .
'''
}
]
} // register : start_addr
{ name: "FATAL_ALERT_CAUSE",
desc: '''
The cause of a fatal alert.
The bits of this register correspond to errors that can cause a fatal
alert. Software can read these bits to see what went wrong. Once set,
these bits cannot be cleared.
'''
swaccess: "ro",
hwaccess: "hwo",
fields: [
{ bits: "0",
name: "imem_error",
resval: 0,
desc: "Set on any ECC error in IMEM"
}
{ bits: "1",
name: "dmem_error",
resval: 0,
desc: "Set on any ECC error in DMEM"
}
{ bits: "2",
name: "reg_error",
resval: 0,
desc: "Set on any ECC error in a register file"
}
]
}
// Give IMEM and DMEM 16 KiB address space, each, to allow for easy expansion
// of the actual IMEM and DMEM sizes without changing the address map.
{ skipto: "0x4000" }
// Imem size (given as `items` below) must be a power of two.
{ window: {
name: "IMEM"
items: "1024" // 4 kB
swaccess: "rw",
byte-write: "false",
desc: '''
Instruction Memory.
This register should only be accesed while OTBN is not busy, as indicated by the !!STATUS.busy flag.
Accesses while OTBN is busy are blocking.
TODO: The exact behavior is yet to be determined, see https://github.com/lowRISC/opentitan/issues/2696 for details.
'''
}
}
{ skipto: "0x8000" }
// Dmem size (given as `items` below) must be a power of two.
{ window: {
name: "DMEM"
items: "1024" // 4 kB
swaccess: "rw",
byte-write: "false",
desc: '''
Data Memory.
This register should only be accesed while OTBN is not busy, as indicated by the !!STATUS.busy flag.
Accesses while OTBN is busy are blocking.
TODO: The exact behavior is yet to be determined, see https://github.com/lowRISC/opentitan/issues/2696 for details.
'''
}
}
]
}