2019-02-06 14:19:48 -06:00
/*
* yosys - - Yosys Open SYnthesis Suite
*
* Copyright ( C ) 2012 Clifford Wolf < clifford @ clifford . at >
2019-02-21 16:40:13 -06:00
* Copyright ( C ) 2019 Eddie Hung < eddie @ fpgeh . com >
2019-02-06 14:19:48 -06:00
*
* Permission to use , copy , modify , and / or distribute this software for any
* purpose with or without fee is hereby granted , provided that the above
* copyright notice and this permission notice appear in all copies .
*
* THE SOFTWARE IS PROVIDED " AS IS " AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS . IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL , DIRECT , INDIRECT , OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE , DATA OR PROFITS , WHETHER IN AN
* ACTION OF CONTRACT , NEGLIGENCE OR OTHER TORTIOUS ACTION , ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE .
*
*/
// [[CITE]] The AIGER And-Inverter Graph (AIG) Format Version 20071012
// Armin Biere. The AIGER And-Inverter Graph (AIG) Format Version 20071012. Technical Report 07/1, October 2011, FMV Reports Series, Institute for Formal Models and Verification, Johannes Kepler University, Altenbergerstr. 69, 4040 Linz, Austria.
// http://fmv.jku.at/papers/Biere-FMV-TR-07-1.pdf
2019-02-19 12:19:53 -06:00
# ifdef _WIN32
2019-02-17 22:59:53 -06:00
# include <libgen.h>
2019-02-19 12:19:53 -06:00
# include <stdlib.h>
# endif
2019-02-17 22:59:53 -06:00
# include <array>
2019-02-06 14:19:48 -06:00
# include "kernel/yosys.h"
# include "kernel/sigtools.h"
2019-02-13 19:00:00 -06:00
# include "kernel/consteval.h"
2019-02-06 14:19:48 -06:00
# include "aigerparse.h"
YOSYS_NAMESPACE_BEGIN
2019-02-12 14:58:10 -06:00
AigerReader : : AigerReader ( RTLIL : : Design * design , std : : istream & f , RTLIL : : IdString module_name , RTLIL : : IdString clk_name , std : : string map_filename , bool wideports )
: design ( design ) , f ( f ) , clk_name ( clk_name ) , map_filename ( map_filename ) , wideports ( wideports )
2019-02-06 14:19:48 -06:00
{
2019-02-08 14:04:26 -06:00
module = new RTLIL : : Module ;
2019-02-08 14:40:43 -06:00
module - > name = module_name ;
2019-02-08 13:45:16 -06:00
if ( design - > module ( module - > name ) )
log_error ( " Duplicate definition of module %s! \n " , log_id ( module - > name ) ) ;
2019-02-08 14:04:26 -06:00
}
2019-02-08 13:45:16 -06:00
2019-02-08 14:04:26 -06:00
void AigerReader : : parse_aiger ( )
{
2019-02-06 14:19:48 -06:00
std : : string header ;
f > > header ;
2019-02-08 14:04:26 -06:00
if ( header ! = " aag " & & header ! = " aig " )
log_error ( " Unsupported AIGER file! \n " ) ;
// Parse rest of header
if ( ! ( f > > M > > I > > L > > O > > A ) )
log_error ( " Invalid AIGER header \n " ) ;
// Optional values
B = C = J = F = 0 ;
2019-05-21 20:20:31 -05:00
if ( f . peek ( ) ! = ' ' ) goto end_of_header ;
2019-05-21 19:20:19 -05:00
if ( ! ( f > > B ) ) log_error ( " Invalid AIGER header \n " ) ;
2019-05-21 20:20:31 -05:00
if ( f . peek ( ) ! = ' ' ) goto end_of_header ;
2019-05-21 19:20:19 -05:00
if ( ! ( f > > C ) ) log_error ( " Invalid AIGER header \n " ) ;
2019-05-21 20:20:31 -05:00
if ( f . peek ( ) ! = ' ' ) goto end_of_header ;
2019-05-21 19:20:19 -05:00
if ( ! ( f > > J ) ) log_error ( " Invalid AIGER header \n " ) ;
2019-05-21 20:20:31 -05:00
if ( f . peek ( ) ! = ' ' ) goto end_of_header ;
2019-05-21 19:20:19 -05:00
if ( ! ( f > > F ) ) log_error ( " Invalid AIGER header \n " ) ;
2019-05-21 20:20:31 -05:00
end_of_header :
2019-02-08 14:04:26 -06:00
std : : string line ;
std : : getline ( f , line ) ; // Ignore up to start of next line, as standard
// says anything that follows could be used for
// optional sections
log_debug ( " M=%u I=%u L=%u O=%u A=%u B=%u C=%u J=%u F=%u \n " , M , I , L , O , A , B , C , J , F ) ;
line_count = 1 ;
2019-02-06 16:58:47 -06:00
if ( header = = " aag " )
2019-02-13 19:00:00 -06:00
parse_aiger_ascii ( ) ;
2019-02-08 09:31:04 -06:00
else if ( header = = " aig " )
2019-02-13 19:00:00 -06:00
parse_aiger_binary ( ) ;
2019-02-08 09:31:04 -06:00
else
2019-02-08 14:04:26 -06:00
log_abort ( ) ;
// Parse footer (symbol table, comments, etc.)
unsigned l1 ;
std : : string s ;
for ( int c = f . peek ( ) ; c ! = EOF ; c = f . peek ( ) , + + line_count ) {
if ( c = = ' i ' | | c = = ' l ' | | c = = ' o ' ) {
f . ignore ( 1 ) ;
if ( ! ( f > > l1 > > s ) )
log_error ( " Line %u cannot be interpreted as a symbol entry! \n " , line_count ) ;
if ( ( c = = ' i ' & & l1 > inputs . size ( ) ) | | ( c = = ' l ' & & l1 > latches . size ( ) ) | | ( c = = ' o ' & & l1 > outputs . size ( ) ) )
log_error ( " Line %u has invalid symbol position! \n " , line_count ) ;
RTLIL : : Wire * wire ;
if ( c = = ' i ' ) wire = inputs [ l1 ] ;
else if ( c = = ' l ' ) wire = latches [ l1 ] ;
else if ( c = = ' o ' ) wire = outputs [ l1 ] ;
else log_abort ( ) ;
module - > rename ( wire , stringf ( " \\ %s " , s . c_str ( ) ) ) ;
}
else if ( c = = ' b ' | | c = = ' j ' | | c = = ' f ' ) {
// TODO
}
else if ( c = = ' c ' ) {
f . ignore ( 1 ) ;
if ( f . peek ( ) = = ' \n ' )
break ;
// Else constraint (TODO)
}
else
log_error ( " Line %u: cannot interpret first character '%c'! \n " , line_count , c ) ;
std : : getline ( f , line ) ; // Ignore up to start of next line
}
2019-02-08 13:45:16 -06:00
2019-04-23 17:06:19 -05:00
post_process ( ) ;
2019-02-08 13:45:16 -06:00
}
2019-02-12 11:36:22 -06:00
static uint32_t parse_xaiger_literal ( std : : istream & f )
{
2019-02-19 12:19:53 -06:00
uint32_t l ;
2019-02-16 10:58:25 -06:00
f . read ( reinterpret_cast < char * > ( & l ) , sizeof ( l ) ) ;
if ( f . gcount ( ) ! = sizeof ( l ) )
2019-02-19 12:19:53 -06:00
log_error ( " Offset %ld: unable to read literal! \n " , static_cast < int64_t > ( f . tellg ( ) ) ) ;
2019-02-19 12:24:55 -06:00
// TODO: Don't assume we're on little endian
2019-02-19 12:19:53 -06:00
# ifdef _WIN32
return _byteswap_ulong ( l ) ;
# else
return __builtin_bswap32 ( l ) ;
# endif
2019-02-12 11:36:22 -06:00
}
2019-02-13 19:00:00 -06:00
static RTLIL : : Wire * createWireIfNotExists ( RTLIL : : Module * module , unsigned literal )
{
const unsigned variable = literal > > 1 ;
const bool invert = literal & 1 ;
2019-02-20 13:22:56 -06:00
RTLIL : : IdString wire_name ( stringf ( " \\ __%d%s__ " , variable , invert ? " b " : " " ) ) ; // FIXME: is "b" the right suffix?
2019-02-13 19:00:00 -06:00
RTLIL : : Wire * wire = module - > wire ( wire_name ) ;
if ( wire ) return wire ;
log_debug ( " Creating %s \n " , wire_name . c_str ( ) ) ;
wire = module - > addWire ( wire_name ) ;
2019-02-15 13:52:05 -06:00
wire - > port_input = wire - > port_output = false ;
2019-02-13 19:00:00 -06:00
if ( ! invert ) return wire ;
2019-02-20 13:22:56 -06:00
RTLIL : : IdString wire_inv_name ( stringf ( " \\ __%d__ " , variable ) ) ;
2019-02-13 19:00:00 -06:00
RTLIL : : Wire * wire_inv = module - > wire ( wire_inv_name ) ;
if ( wire_inv ) {
if ( module - > cell ( wire_inv_name ) ) return wire ;
}
else {
log_debug ( " Creating %s \n " , wire_inv_name . c_str ( ) ) ;
wire_inv = module - > addWire ( wire_inv_name ) ;
2019-02-15 13:52:05 -06:00
wire_inv - > port_input = wire_inv - > port_output = false ;
2019-02-13 19:00:00 -06:00
}
log_debug ( " Creating %s = ~%s \n " , wire_name . c_str ( ) , wire_inv_name . c_str ( ) ) ;
2019-02-20 18:30:30 -06:00
module - > addNotGate ( stringf ( " \\ __%d__$not " , variable ) , wire_inv , wire ) ; // FIXME: is "$not" the right suffix?
2019-02-13 19:00:00 -06:00
return wire ;
}
2019-02-21 19:01:07 -06:00
static std : : pair < RTLIL : : IdString , int > wideports_split ( std : : string name )
{
int pos = - 1 ;
if ( name . empty ( ) | | name . back ( ) ! = ' ] ' )
goto failed ;
for ( int i = 0 ; i + 1 < GetSize ( name ) ; i + + ) {
if ( name [ i ] = = ' [ ' )
pos = i ;
else if ( name [ i ] < ' 0 ' | | name [ i ] > ' 9 ' )
pos = - 1 ;
else if ( i = = pos + 1 & & name [ i ] = = ' 0 ' & & name [ i + 1 ] ! = ' ] ' )
pos = - 1 ;
}
if ( pos > = 0 )
return std : : pair < RTLIL : : IdString , int > ( RTLIL : : escape_id ( name . substr ( 0 , pos ) ) , atoi ( name . c_str ( ) + pos + 1 ) ) ;
failed :
return std : : pair < RTLIL : : IdString , int > ( name , 0 ) ;
}
2019-02-11 17:19:17 -06:00
void AigerReader : : parse_xaiger ( )
{
std : : string header ;
f > > header ;
if ( header ! = " aag " & & header ! = " aig " )
log_error ( " Unsupported AIGER file! \n " ) ;
// Parse rest of header
if ( ! ( f > > M > > I > > L > > O > > A ) )
log_error ( " Invalid AIGER header \n " ) ;
// Optional values
B = C = J = F = 0 ;
std : : string line ;
std : : getline ( f , line ) ; // Ignore up to start of next line, as standard
// says anything that follows could be used for
// optional sections
log_debug ( " M=%u I=%u L=%u O=%u A=%u \n " , M , I , L , O , A ) ;
line_count = 1 ;
if ( header = = " aag " )
2019-02-13 19:00:00 -06:00
parse_aiger_ascii ( ) ;
2019-02-11 17:19:17 -06:00
else if ( header = = " aig " )
2019-02-13 19:00:00 -06:00
parse_aiger_binary ( ) ;
2019-02-11 17:19:17 -06:00
else
log_abort ( ) ;
// Parse footer (symbol table, comments, etc.)
unsigned l1 ;
std : : string s ;
2019-02-12 11:36:22 -06:00
bool comment_seen = false ;
2019-02-21 19:01:07 -06:00
std : : vector < std : : pair < RTLIL : : Wire * , RTLIL : : IdString > > deferred_renames ;
2019-02-25 20:40:23 -06:00
std : : vector < std : : pair < RTLIL : : Wire * , RTLIL : : IdString > > deferred_inouts ;
2019-02-21 19:01:07 -06:00
deferred_renames . reserve ( inputs . size ( ) + latches . size ( ) + outputs . size ( ) ) ;
2019-02-11 17:19:17 -06:00
for ( int c = f . peek ( ) ; c ! = EOF ; c = f . peek ( ) ) {
2019-02-12 11:36:22 -06:00
if ( comment_seen | | c = = ' c ' ) {
if ( ! comment_seen ) {
f . ignore ( 1 ) ;
c = f . peek ( ) ;
comment_seen = true ;
}
2019-02-13 19:00:00 -06:00
if ( c = = ' \n ' )
break ;
f . ignore ( 1 ) ;
2019-02-12 11:36:22 -06:00
// XAIGER extensions
if ( c = = ' m ' ) {
uint32_t dataSize = parse_xaiger_literal ( f ) ;
uint32_t lutNum = parse_xaiger_literal ( f ) ;
uint32_t lutSize = parse_xaiger_literal ( f ) ;
log_debug ( " m: dataSize=%u lutNum=%u lutSize=%u \n " , dataSize , lutNum , lutSize ) ;
2019-02-17 00:22:29 -06:00
ConstEval ce ( module ) ;
2019-02-12 11:36:22 -06:00
for ( unsigned i = 0 ; i < lutNum ; + + i ) {
uint32_t rootNodeID = parse_xaiger_literal ( f ) ;
uint32_t cutLeavesM = parse_xaiger_literal ( f ) ;
log_debug ( " rootNodeID=%d cutLeavesM=%d \n " , rootNodeID , cutLeavesM ) ;
2019-02-20 13:22:56 -06:00
RTLIL : : Wire * output_sig = module - > wire ( stringf ( " \\ __%d__ " , rootNodeID ) ) ;
2019-02-12 11:36:22 -06:00
uint32_t nodeID ;
RTLIL : : SigSpec input_sig ;
for ( unsigned j = 0 ; j < cutLeavesM ; + + j ) {
nodeID = parse_xaiger_literal ( f ) ;
log_debug ( " \t %u \n " , nodeID ) ;
2019-02-20 13:22:56 -06:00
RTLIL : : Wire * wire = module - > wire ( stringf ( " \\ __%d__ " , nodeID ) ) ;
2019-02-12 11:36:22 -06:00
log_assert ( wire ) ;
input_sig . append ( wire ) ;
}
2019-02-13 19:00:00 -06:00
RTLIL : : Const lut_mask ( RTLIL : : State : : Sx , 1 < < input_sig . size ( ) ) ;
for ( int j = 0 ; j < ( 1 < < cutLeavesM ) ; + + j ) {
ce . push ( ) ;
ce . set ( input_sig , RTLIL : : Const { j , static_cast < int > ( cutLeavesM ) } ) ;
RTLIL : : SigSpec o ( output_sig ) ;
ce . eval ( o ) ;
lut_mask [ j ] = o . as_const ( ) [ 0 ] ;
ce . pop ( ) ;
}
2019-02-20 18:30:30 -06:00
RTLIL : : Cell * output_cell = module - > cell ( stringf ( " \\ __%d__$and " , rootNodeID ) ) ;
2019-02-13 19:00:00 -06:00
log_assert ( output_cell ) ;
module - > remove ( output_cell ) ;
2019-04-10 16:02:23 -05:00
module - > addLut ( stringf ( " \\ __%d__$lut " , rootNodeID ) , input_sig , output_sig , std : : move ( lut_mask ) ) ;
2019-02-12 11:36:22 -06:00
}
}
else if ( c = = ' n ' ) {
2019-02-13 19:00:00 -06:00
parse_xaiger_literal ( f ) ;
f > > s ;
log_debug ( " n: '%s' \n " , s . c_str ( ) ) ;
2019-02-12 11:36:22 -06:00
}
2019-04-17 12:55:23 -05:00
else if ( c = = ' a ' | | c = = ' i ' | | c = = ' o ' | | c = = ' h ' ) {
uint32_t dataSize = parse_xaiger_literal ( f ) ;
f . ignore ( dataSize ) ;
}
else {
break ;
}
2019-02-12 11:36:22 -06:00
}
else if ( c = = ' i ' | | c = = ' l ' | | c = = ' o ' ) {
2019-02-11 17:19:17 -06:00
f . ignore ( 1 ) ;
if ( ! ( f > > l1 > > s ) )
log_error ( " Line %u cannot be interpreted as a symbol entry! \n " , line_count ) ;
if ( ( c = = ' i ' & & l1 > inputs . size ( ) ) | | ( c = = ' l ' & & l1 > latches . size ( ) ) | | ( c = = ' o ' & & l1 > outputs . size ( ) ) )
log_error ( " Line %u has invalid symbol position! \n " , line_count ) ;
RTLIL : : Wire * wire ;
if ( c = = ' i ' ) wire = inputs [ l1 ] ;
else if ( c = = ' l ' ) wire = latches [ l1 ] ;
else if ( c = = ' o ' ) wire = outputs [ l1 ] ;
else log_abort ( ) ;
2019-02-26 14:04:45 -06:00
RTLIL : : IdString escaped_s = RTLIL : : escape_id ( s ) ;
if ( escaped_s . ends_with ( " $inout.out " ) ) {
deferred_inouts . emplace_back ( wire , escaped_s . substr ( 0 , escaped_s . size ( ) - 10 ) ) ;
goto next_line ;
}
else if ( wideports & & ( wire - > port_input | | wire - > port_output ) ) {
RTLIL : : IdString wide_symbol ;
2019-02-26 13:37:34 -06:00
int index ;
2019-02-26 14:04:45 -06:00
std : : tie ( wide_symbol , index ) = wideports_split ( escaped_s . str ( ) ) ;
if ( wide_symbol . ends_with ( " $inout.out " ) ) {
deferred_inouts . emplace_back ( wire , stringf ( " %s[%d] " , wide_symbol . substr ( 0 , wide_symbol . size ( ) - 10 ) . c_str ( ) , index ) ) ;
2019-02-26 13:37:34 -06:00
goto next_line ;
}
}
2019-02-26 14:04:45 -06:00
deferred_renames . emplace_back ( wire , escaped_s ) ;
2019-02-19 14:27:50 -06:00
2019-02-26 13:37:34 -06:00
next_line :
2019-02-12 11:36:22 -06:00
std : : getline ( f , line ) ; // Ignore up to start of next line
+ + line_count ;
2019-02-11 17:19:17 -06:00
}
else
log_error ( " Line %u: cannot interpret first character '%c'! \n " , line_count , c ) ;
}
2019-02-12 14:16:37 -06:00
dict < RTLIL : : IdString , int > wideports_cache ;
2019-02-25 20:40:23 -06:00
for ( const auto & i : deferred_renames ) {
2019-02-21 19:01:07 -06:00
RTLIL : : Wire * wire = i . first ;
module - > rename ( wire , i . second ) ;
if ( wideports & & ( wire - > port_input | | wire - > port_output ) ) {
RTLIL : : IdString escaped_symbol ;
int index ;
std : : tie ( escaped_symbol , index ) = wideports_split ( wire - > name . str ( ) ) ;
if ( index > 0 )
wideports_cache [ escaped_symbol ] = std : : max ( wideports_cache [ escaped_symbol ] , index ) ;
}
}
2019-02-12 14:16:37 -06:00
2019-02-25 20:40:23 -06:00
for ( const auto & i : deferred_inouts ) {
RTLIL : : Wire * out_wire = i . first ;
log_assert ( out_wire - > port_output ) ;
out_wire - > port_output = false ;
RTLIL : : Wire * wire = module - > wire ( i . second ) ;
log_assert ( wire ) ;
log_assert ( wire - > port_input & & ! wire - > port_output ) ;
wire - > port_output = true ;
module - > connect ( wire , out_wire ) ;
}
2019-04-23 17:06:19 -05:00
post_process ( ) ;
2019-02-11 17:19:17 -06:00
}
2019-02-13 19:00:00 -06:00
void AigerReader : : parse_aiger_ascii ( )
2019-02-06 16:58:47 -06:00
{
2019-02-08 12:54:31 -06:00
std : : string line ;
2019-02-06 14:19:48 -06:00
std : : stringstream ss ;
2019-02-08 10:09:30 -06:00
unsigned l1 , l2 , l3 ;
2019-02-06 14:19:48 -06:00
// Parse inputs
2019-02-08 12:54:31 -06:00
for ( unsigned i = 0 ; i < I ; + + i , + + line_count ) {
2019-02-08 10:04:48 -06:00
if ( ! ( f > > l1 ) )
2019-02-08 12:54:31 -06:00
log_error ( " Line %u cannot be interpreted as an input! \n " , line_count ) ;
2019-02-06 14:19:48 -06:00
log_debug ( " %d is an input \n " , l1 ) ;
2019-02-25 20:40:23 -06:00
log_assert ( ! ( l1 & 1 ) ) ; // Inputs can't be inverted
2019-02-08 13:45:16 -06:00
RTLIL : : Wire * wire = createWireIfNotExists ( module , l1 ) ;
2019-02-06 14:19:48 -06:00
wire - > port_input = true ;
2019-02-08 10:03:40 -06:00
inputs . push_back ( wire ) ;
2019-02-06 14:19:48 -06:00
}
// Parse latches
2019-02-08 10:08:49 -06:00
RTLIL : : Wire * clk_wire = nullptr ;
if ( L > 0 ) {
2019-02-25 17:34:02 -06:00
log_assert ( clk_name ! = " " ) ;
2019-02-08 14:40:43 -06:00
clk_wire = module - > wire ( clk_name ) ;
2019-02-08 10:08:49 -06:00
log_assert ( ! clk_wire ) ;
2019-02-08 14:40:43 -06:00
log_debug ( " Creating %s \n " , clk_name . c_str ( ) ) ;
clk_wire = module - > addWire ( clk_name ) ;
2019-02-08 10:08:49 -06:00
clk_wire - > port_input = true ;
2019-02-15 13:52:05 -06:00
clk_wire - > port_output = false ;
2019-02-08 10:08:49 -06:00
}
2019-02-08 12:54:31 -06:00
for ( unsigned i = 0 ; i < L ; + + i , + + line_count ) {
2019-02-08 10:04:48 -06:00
if ( ! ( f > > l1 > > l2 ) )
2019-02-08 12:54:31 -06:00
log_error ( " Line %u cannot be interpreted as a latch! \n " , line_count ) ;
2019-02-06 14:19:48 -06:00
log_debug ( " %d %d is a latch \n " , l1 , l2 ) ;
log_assert ( ! ( l1 & 1 ) ) ; // TODO: Latch outputs can't be inverted?
2019-02-08 13:45:16 -06:00
RTLIL : : Wire * q_wire = createWireIfNotExists ( module , l1 ) ;
RTLIL : : Wire * d_wire = createWireIfNotExists ( module , l2 ) ;
2019-02-06 14:19:48 -06:00
2019-02-08 15:17:53 -06:00
module - > addDffGate ( NEW_ID , clk_wire , d_wire , q_wire ) ;
2019-02-08 10:37:18 -06:00
2019-02-08 10:37:44 -06:00
// Reset logic is optional in AIGER 1.9
2019-02-08 10:37:18 -06:00
if ( f . peek ( ) = = ' ' ) {
if ( ! ( f > > l3 ) )
2019-02-08 12:54:31 -06:00
log_error ( " Line %u cannot be interpreted as a latch! \n " , line_count ) ;
2019-02-08 10:37:18 -06:00
if ( l3 = = 0 | | l3 = = 1 )
2019-02-08 13:45:16 -06:00
q_wire - > attributes [ " \\ init " ] = RTLIL : : Const ( l3 ) ;
2019-02-08 10:37:18 -06:00
else if ( l3 = = l1 ) {
//q_wire->attributes["\\init"] = RTLIL::Const(RTLIL::State::Sx);
}
else
2019-02-08 12:54:31 -06:00
log_error ( " Line %u has invalid reset literal for latch! \n " , line_count ) ;
2019-02-08 10:37:18 -06:00
}
else {
// AIGER latches are assumed to be initialized to zero
q_wire - > attributes [ " \\ init " ] = RTLIL : : Const ( 0 ) ;
}
2019-02-08 10:03:40 -06:00
latches . push_back ( q_wire ) ;
2019-02-06 14:19:48 -06:00
}
// Parse outputs
2019-02-08 12:54:31 -06:00
for ( unsigned i = 0 ; i < O ; + + i , + + line_count ) {
2019-02-08 10:04:48 -06:00
if ( ! ( f > > l1 ) )
2019-02-08 12:54:31 -06:00
log_error ( " Line %u cannot be interpreted as an output! \n " , line_count ) ;
2019-02-06 14:19:48 -06:00
2019-02-16 10:44:11 -06:00
RTLIL : : Wire * wire ;
if ( l1 = = 0 | | l1 = = 1 ) {
2019-02-20 13:22:56 -06:00
wire = module - > addWire ( NEW_ID ) ;
2019-02-16 10:44:11 -06:00
if ( l1 = = 0 )
module - > connect ( wire , RTLIL : : State : : S0 ) ;
else if ( l1 = = 1 )
module - > connect ( wire , RTLIL : : State : : S1 ) ;
else
log_abort ( ) ;
}
else {
log_debug ( " %d is an output \n " , l1 ) ;
2019-02-19 17:15:50 -06:00
const unsigned variable = l1 > > 1 ;
const bool invert = l1 & 1 ;
2019-02-20 13:22:56 -06:00
RTLIL : : IdString wire_name ( stringf ( " \\ __%d%s__ " , variable , invert ? " b " : " " ) ) ; // FIXME: is "b" the right suffix?
2019-02-19 17:15:50 -06:00
wire = module - > wire ( wire_name ) ;
if ( ! wire )
wire = createWireIfNotExists ( module , l1 ) ;
else {
2019-02-25 20:40:23 -06:00
if ( wire - > port_input | | wire - > port_output ) {
2019-02-20 13:22:56 -06:00
RTLIL : : Wire * new_wire = module - > addWire ( NEW_ID ) ;
2019-02-19 17:15:50 -06:00
module - > connect ( new_wire , wire ) ;
wire = new_wire ;
}
}
2019-02-16 23:53:03 -06:00
}
2019-02-06 14:19:48 -06:00
wire - > port_output = true ;
2019-02-08 10:03:40 -06:00
outputs . push_back ( wire ) ;
2019-02-06 14:19:48 -06:00
}
std : : getline ( f , line ) ; // Ignore up to start of next line
2019-02-08 10:03:40 -06:00
2019-02-06 14:19:48 -06:00
// TODO: Parse bad state properties
2019-02-08 12:54:31 -06:00
for ( unsigned i = 0 ; i < B ; + + i , + + line_count )
2019-02-06 14:19:48 -06:00
std : : getline ( f , line ) ; // Ignore up to start of next line
// TODO: Parse invariant constraints
2019-02-08 12:54:31 -06:00
for ( unsigned i = 0 ; i < C ; + + i , + + line_count )
2019-02-06 14:19:48 -06:00
std : : getline ( f , line ) ; // Ignore up to start of next line
// TODO: Parse justice properties
2019-02-08 12:54:31 -06:00
for ( unsigned i = 0 ; i < J ; + + i , + + line_count )
2019-02-06 14:19:48 -06:00
std : : getline ( f , line ) ; // Ignore up to start of next line
// TODO: Parse fairness constraints
2019-02-08 12:54:31 -06:00
for ( unsigned i = 0 ; i < F ; + + i , + + line_count )
2019-02-06 14:19:48 -06:00
std : : getline ( f , line ) ; // Ignore up to start of next line
// Parse AND
2019-02-11 15:24:21 -06:00
for ( unsigned i = 0 ; i < A ; + + i ) {
2019-02-08 10:04:48 -06:00
if ( ! ( f > > l1 > > l2 > > l3 ) )
2019-02-08 12:54:31 -06:00
log_error ( " Line %u cannot be interpreted as an AND! \n " , line_count ) ;
2019-02-06 14:19:48 -06:00
log_debug ( " %d %d %d is an AND \n " , l1 , l2 , l3 ) ;
2019-02-13 19:00:00 -06:00
log_assert ( ! ( l1 & 1 ) ) ;
RTLIL : : Wire * o_wire = createWireIfNotExists ( module , l1 ) ;
RTLIL : : Wire * i1_wire = createWireIfNotExists ( module , l2 ) ;
RTLIL : : Wire * i2_wire = createWireIfNotExists ( module , l3 ) ;
2019-02-20 18:30:30 -06:00
module - > addAndGate ( o_wire - > name . str ( ) + " $and " , i1_wire , i2_wire , o_wire ) ;
2019-02-06 14:19:48 -06:00
}
2019-02-17 14:07:14 -06:00
std : : getline ( f , line ) ; // Ignore up to start of next line
2019-02-08 13:45:16 -06:00
}
2019-02-06 14:19:48 -06:00
2019-02-08 13:45:16 -06:00
static unsigned parse_next_delta_literal ( std : : istream & f , unsigned ref )
{
unsigned x = 0 , i = 0 ;
unsigned char ch ;
while ( ( ch = f . get ( ) ) & 0x80 )
x | = ( ch & 0x7f ) < < ( 7 * i + + ) ;
return ref - ( x | ( ch < < ( 7 * i ) ) ) ;
2019-02-06 14:19:48 -06:00
}
2019-02-13 19:00:00 -06:00
void AigerReader : : parse_aiger_binary ( )
2019-02-08 09:31:04 -06:00
{
2019-02-08 13:45:16 -06:00
unsigned l1 , l2 , l3 ;
std : : string line ;
// Parse inputs
for ( unsigned i = 1 ; i < = I ; + + i ) {
2019-02-12 11:36:22 -06:00
log_debug ( " %d is an input \n " , i ) ;
2019-02-08 13:45:16 -06:00
RTLIL : : Wire * wire = createWireIfNotExists ( module , i < < 1 ) ;
wire - > port_input = true ;
2019-02-15 13:52:05 -06:00
log_assert ( ! wire - > port_output ) ;
2019-02-08 13:45:16 -06:00
inputs . push_back ( wire ) ;
}
// Parse latches
RTLIL : : Wire * clk_wire = nullptr ;
if ( L > 0 ) {
2019-02-25 17:34:02 -06:00
log_assert ( clk_name ! = " " ) ;
2019-02-08 14:40:43 -06:00
clk_wire = module - > wire ( clk_name ) ;
2019-02-08 13:45:16 -06:00
log_assert ( ! clk_wire ) ;
2019-02-08 14:40:43 -06:00
log_debug ( " Creating %s \n " , clk_name . c_str ( ) ) ;
clk_wire = module - > addWire ( clk_name ) ;
2019-02-08 13:45:16 -06:00
clk_wire - > port_input = true ;
2019-02-15 13:52:05 -06:00
clk_wire - > port_output = false ;
2019-02-08 13:45:16 -06:00
}
l1 = ( I + 1 ) * 2 ;
for ( unsigned i = 0 ; i < L ; + + i , + + line_count , l1 + = 2 ) {
if ( ! ( f > > l2 ) )
log_error ( " Line %u cannot be interpreted as a latch! \n " , line_count ) ;
log_debug ( " %d %d is a latch \n " , l1 , l2 ) ;
RTLIL : : Wire * q_wire = createWireIfNotExists ( module , l1 ) ;
RTLIL : : Wire * d_wire = createWireIfNotExists ( module , l2 ) ;
module - > addDff ( NEW_ID , clk_wire , d_wire , q_wire ) ;
// Reset logic is optional in AIGER 1.9
if ( f . peek ( ) = = ' ' ) {
if ( ! ( f > > l3 ) )
log_error ( " Line %u cannot be interpreted as a latch! \n " , line_count ) ;
if ( l3 = = 0 | | l3 = = 1 )
q_wire - > attributes [ " \\ init " ] = RTLIL : : Const ( l3 ) ;
else if ( l3 = = l1 ) {
//q_wire->attributes["\\init"] = RTLIL::Const(RTLIL::State::Sx);
}
else
log_error ( " Line %u has invalid reset literal for latch! \n " , line_count ) ;
}
else {
// AIGER latches are assumed to be initialized to zero
q_wire - > attributes [ " \\ init " ] = RTLIL : : Const ( 0 ) ;
}
latches . push_back ( q_wire ) ;
}
// Parse outputs
for ( unsigned i = 0 ; i < O ; + + i , + + line_count ) {
if ( ! ( f > > l1 ) )
log_error ( " Line %u cannot be interpreted as an output! \n " , line_count ) ;
2019-02-16 10:44:11 -06:00
RTLIL : : Wire * wire ;
if ( l1 = = 0 | | l1 = = 1 ) {
2019-02-20 13:22:56 -06:00
wire = module - > addWire ( NEW_ID ) ;
2019-02-16 10:44:11 -06:00
if ( l1 = = 0 )
module - > connect ( wire , RTLIL : : State : : S0 ) ;
else if ( l1 = = 1 )
module - > connect ( wire , RTLIL : : State : : S1 ) ;
else
log_abort ( ) ;
}
else {
log_debug ( " %d is an output \n " , l1 ) ;
2019-02-19 17:14:08 -06:00
const unsigned variable = l1 > > 1 ;
const bool invert = l1 & 1 ;
2019-02-20 14:39:51 -06:00
RTLIL : : IdString wire_name ( stringf ( " \\ __%d%s__ " , variable , invert ? " b " : " " ) ) ; // FIXME: is "_inv" the right suffix?
2019-02-19 17:14:08 -06:00
wire = module - > wire ( wire_name ) ;
if ( ! wire )
wire = createWireIfNotExists ( module , l1 ) ;
else {
2019-02-25 20:40:23 -06:00
if ( wire - > port_input | | wire - > port_output ) {
2019-02-20 13:22:56 -06:00
RTLIL : : Wire * new_wire = module - > addWire ( NEW_ID ) ;
2019-02-19 17:14:08 -06:00
module - > connect ( new_wire , wire ) ;
wire = new_wire ;
}
}
2019-02-16 23:53:03 -06:00
}
2019-02-16 15:45:51 -06:00
wire - > port_output = true ;
2019-02-08 13:45:16 -06:00
outputs . push_back ( wire ) ;
}
std : : getline ( f , line ) ; // Ignore up to start of next line
// TODO: Parse bad state properties
for ( unsigned i = 0 ; i < B ; + + i , + + line_count )
std : : getline ( f , line ) ; // Ignore up to start of next line
// TODO: Parse invariant constraints
for ( unsigned i = 0 ; i < C ; + + i , + + line_count )
std : : getline ( f , line ) ; // Ignore up to start of next line
// TODO: Parse justice properties
for ( unsigned i = 0 ; i < J ; + + i , + + line_count )
std : : getline ( f , line ) ; // Ignore up to start of next line
// TODO: Parse fairness constraints
for ( unsigned i = 0 ; i < F ; + + i , + + line_count )
std : : getline ( f , line ) ; // Ignore up to start of next line
// Parse AND
l1 = ( I + L + 1 ) < < 1 ;
for ( unsigned i = 0 ; i < A ; + + i , + + line_count , l1 + = 2 ) {
l2 = parse_next_delta_literal ( f , l1 ) ;
l3 = parse_next_delta_literal ( f , l2 ) ;
log_debug ( " %d %d %d is an AND \n " , l1 , l2 , l3 ) ;
2019-02-13 19:00:00 -06:00
log_assert ( ! ( l1 & 1 ) ) ;
RTLIL : : Wire * o_wire = createWireIfNotExists ( module , l1 ) ;
RTLIL : : Wire * i1_wire = createWireIfNotExists ( module , l2 ) ;
RTLIL : : Wire * i2_wire = createWireIfNotExists ( module , l3 ) ;
2019-02-20 18:30:30 -06:00
module - > addAndGate ( o_wire - > name . str ( ) + " $and " , i1_wire , i2_wire , o_wire ) ;
2019-02-08 13:45:16 -06:00
}
2019-02-08 09:31:04 -06:00
}
2019-04-23 17:06:19 -05:00
void AigerReader : : post_process ( )
{
dict < RTLIL : : IdString , int > wideports_cache ;
if ( ! map_filename . empty ( ) ) {
std : : ifstream mf ( map_filename ) ;
std : : string type , symbol ;
int variable , index ;
while ( mf > > type > > variable > > index > > symbol ) {
RTLIL : : IdString escaped_s = RTLIL : : escape_id ( symbol ) ;
if ( type = = " input " ) {
log_assert ( static_cast < unsigned > ( variable ) < inputs . size ( ) ) ;
RTLIL : : Wire * wire = inputs [ variable ] ;
log_assert ( wire ) ;
log_assert ( wire - > port_input ) ;
if ( index = = 0 ) {
// Cope with the fact that a CI might be identical
// to a PI (necessary due to ABC); in those cases
// simply connect the latter to the former
RTLIL : : Wire * existing = module - > wire ( escaped_s ) ;
if ( ! existing )
module - > rename ( wire , escaped_s ) ;
else {
wire - > port_input = false ;
module - > connect ( wire , existing ) ;
}
}
else if ( index > 0 ) {
std : : string indexed_name = stringf ( " %s[%d] " , escaped_s . c_str ( ) , index ) ;
RTLIL : : Wire * existing = module - > wire ( indexed_name ) ;
if ( ! existing ) {
module - > rename ( wire , indexed_name ) ;
if ( wideports )
wideports_cache [ escaped_s ] = std : : max ( wideports_cache [ escaped_s ] , index ) ;
}
else {
module - > connect ( wire , existing ) ;
wire - > port_input = false ;
}
}
}
else if ( type = = " output " ) {
log_assert ( static_cast < unsigned > ( variable ) < outputs . size ( ) ) ;
RTLIL : : Wire * wire = outputs [ variable ] ;
log_assert ( wire ) ;
log_assert ( wire - > port_output ) ;
if ( escaped_s . in ( " \\ __dummy_o__ " , " \\ __const0__ " , " \\ __const1__ " ) ) {
wire - > port_output = false ;
continue ;
}
if ( index = = 0 ) {
// Cope with the fact that a CO might be identical
// to a PO (necessary due to ABC); in those cases
// simply connect the latter to the former
RTLIL : : Wire * existing = module - > wire ( escaped_s ) ;
if ( ! existing ) {
if ( escaped_s . ends_with ( " $inout.out " ) ) {
wire - > port_output = false ;
RTLIL : : Wire * in_wire = module - > wire ( escaped_s . substr ( 0 , escaped_s . size ( ) - 10 ) ) ;
log_assert ( in_wire ) ;
log_assert ( in_wire - > port_input & & ! in_wire - > port_output ) ;
in_wire - > port_output = true ;
module - > connect ( in_wire , wire ) ;
}
else
module - > rename ( wire , escaped_s ) ;
}
else {
wire - > port_output = false ;
module - > connect ( wire , existing ) ;
}
}
else if ( index > 0 ) {
std : : string indexed_name = stringf ( " %s[%d] " , escaped_s . c_str ( ) , index ) ;
RTLIL : : Wire * existing = module - > wire ( indexed_name ) ;
if ( ! existing ) {
if ( escaped_s . ends_with ( " $inout.out " ) ) {
wire - > port_output = false ;
RTLIL : : Wire * in_wire = module - > wire ( stringf ( " %s[%d] " , escaped_s . substr ( 0 , escaped_s . size ( ) - 10 ) . c_str ( ) , index ) ) ;
log_assert ( in_wire ) ;
log_assert ( in_wire - > port_input & & ! in_wire - > port_output ) ;
in_wire - > port_output = true ;
module - > connect ( in_wire , wire ) ;
}
else {
module - > rename ( wire , indexed_name ) ;
if ( wideports )
wideports_cache [ escaped_s ] = std : : max ( wideports_cache [ escaped_s ] , index ) ;
}
}
else {
module - > connect ( wire , existing ) ;
wire - > port_output = false ;
}
}
}
2019-05-27 13:38:52 -05:00
else if ( type = = " cinput " | | type = = " coutput " ) {
RTLIL : : Wire * wire ;
if ( type = = " cinput " ) {
log_assert ( static_cast < unsigned > ( variable ) < inputs . size ( ) ) ;
wire = inputs [ variable ] ;
log_assert ( wire ) ;
log_assert ( wire - > port_input ) ;
}
else if ( type = = " coutput " ) {
log_assert ( static_cast < unsigned > ( variable ) < outputs . size ( ) ) ;
wire = outputs [ variable ] ;
log_assert ( wire ) ;
log_assert ( wire - > port_output ) ;
}
else log_abort ( ) ;
std : : string port ;
mf > > port ;
RTLIL : : IdString cell_name = RTLIL : : escape_id ( symbol ) ;
RTLIL : : IdString cell_port = RTLIL : : escape_id ( port ) ;
RTLIL : : Cell * cell = module - > cell ( cell_name ) ;
if ( ! cell )
cell = module - > addCell ( cell_name , " $__blackbox__ " ) ;
wire - > port_input = false ;
wire - > port_output = false ;
if ( cell - > hasPort ( cell_port ) ) {
log_assert ( index = = GetSize ( cell - > getPort ( cell_port ) ) ) ;
cell - > connections_ [ cell_port ] . append ( wire ) ;
}
else {
log_assert ( index = = 0 ) ;
cell - > setPort ( cell_port , wire ) ;
}
}
2019-04-23 17:06:19 -05:00
else
log_error ( " Symbol type '%s' not recognised. \n " , type . c_str ( ) ) ;
}
}
for ( auto & wp : wideports_cache ) {
auto name = wp . first ;
int width = wp . second + 1 ;
RTLIL : : Wire * wire = module - > wire ( name ) ;
if ( wire )
module - > rename ( wire , RTLIL : : escape_id ( stringf ( " %s[%d] " , name . c_str ( ) , 0 ) ) ) ;
// Do not make ports with a mix of input/output into
// wide ports
bool port_input = false , port_output = false ;
for ( int i = 0 ; i < width ; i + + ) {
RTLIL : : IdString other_name = name . str ( ) + stringf ( " [%d] " , i ) ;
RTLIL : : Wire * other_wire = module - > wire ( other_name ) ;
if ( other_wire ) {
port_input = port_input | | other_wire - > port_input ;
port_output = port_output | | other_wire - > port_output ;
}
}
if ( ( port_input & & port_output ) | | ( ! port_input & & ! port_output ) )
continue ;
wire = module - > addWire ( name , width ) ;
wire - > port_input = port_input ;
wire - > port_output = port_output ;
for ( int i = 0 ; i < width ; i + + ) {
RTLIL : : IdString other_name = name . str ( ) + stringf ( " [%d] " , i ) ;
RTLIL : : Wire * other_wire = module - > wire ( other_name ) ;
if ( other_wire ) {
other_wire - > port_input = false ;
other_wire - > port_output = false ;
if ( wire - > port_input )
module - > connect ( other_wire , SigSpec ( wire , i ) ) ;
else
module - > connect ( SigSpec ( wire , i ) , other_wire ) ;
}
}
}
module - > fixup_ports ( ) ;
design - > add ( module ) ;
Pass : : call ( design , " clean " ) ;
for ( auto cell : module - > cells ( ) . to_vector ( ) ) {
if ( cell - > type ! = " $lut " ) continue ;
auto y_port = cell - > getPort ( " \\ Y " ) . as_bit ( ) ;
if ( y_port . wire - > width = = 1 )
module - > rename ( cell , stringf ( " %s$lut " , y_port . wire - > name . c_str ( ) ) ) ;
else
module - > rename ( cell , stringf ( " %s[%d]$lut " , y_port . wire - > name . c_str ( ) , y_port . offset ) ) ;
}
}
2019-02-06 14:19:48 -06:00
struct AigerFrontend : public Frontend {
2019-02-08 15:17:02 -06:00
AigerFrontend ( ) : Frontend ( " aiger " , " read AIGER file " ) { }
void help ( ) YS_OVERRIDE
{
// |---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|---v---|
2019-02-08 14:49:55 -06:00
log ( " \n " ) ;
2019-02-08 15:17:02 -06:00
log ( " read_aiger [options] [filename] \n " ) ;
log ( " \n " ) ;
log ( " Load module from an AIGER file into the current design. \n " ) ;
log ( " \n " ) ;
log ( " -module_name <module_name> \n " ) ;
2019-02-19 12:19:53 -06:00
log ( " Name of module to be created (default: <filename>) \n " ) ;
2019-02-08 15:17:02 -06:00
log ( " \n " ) ;
log ( " -clk_name <wire_name> \n " ) ;
log ( " AIGER latches to be transformed into posedge DFFs clocked by wire of " ) ;
log ( " this name (default: clk) \n " ) ;
2019-02-12 14:16:37 -06:00
log ( " \n " ) ;
2019-04-08 18:37:22 -05:00
log ( " -map <filename> \n " ) ;
log ( " read file with port and latch symbols \n " ) ;
log ( " \n " ) ;
log ( " -wideports \n " ) ;
log ( " Merge ports that match the pattern 'name[int]' into a single \n " ) ;
log ( " multi-bit port 'name'. \n " ) ;
2019-02-08 15:17:02 -06:00
log ( " \n " ) ;
}
void execute ( std : : istream * & f , std : : string filename , std : : vector < std : : string > args , RTLIL : : Design * design ) YS_OVERRIDE
{
log_header ( design , " Executing AIGER frontend. \n " ) ;
2019-02-06 14:19:48 -06:00
2019-02-08 14:40:43 -06:00
RTLIL : : IdString clk_name = " \\ clk " ;
RTLIL : : IdString module_name ;
2019-02-12 14:16:37 -06:00
std : : string map_filename ;
2019-02-12 14:58:10 -06:00
bool wideports = false ;
2019-02-08 14:40:43 -06:00
2019-04-08 18:37:22 -05:00
size_t argidx ;
for ( argidx = 1 ; argidx < args . size ( ) ; argidx + + ) {
std : : string arg = args [ argidx ] ;
if ( arg = = " -module_name " & & argidx + 1 < args . size ( ) ) {
module_name = RTLIL : : escape_id ( args [ + + argidx ] ) ;
continue ;
}
if ( arg = = " -clk_name " & & argidx + 1 < args . size ( ) ) {
clk_name = RTLIL : : escape_id ( args [ + + argidx ] ) ;
continue ;
}
if ( map_filename . empty ( ) & & arg = = " -map " & & argidx + 1 < args . size ( ) ) {
map_filename = args [ + + argidx ] ;
continue ;
}
if ( arg = = " -wideports " ) {
wideports = true ;
continue ;
}
break ;
}
extra_args ( f , filename , args , argidx ) ;
2019-02-06 14:19:48 -06:00
2019-02-08 14:40:43 -06:00
if ( module_name . empty ( ) ) {
# ifdef _WIN32
2019-02-19 12:19:53 -06:00
char fname [ _MAX_FNAME ] ;
_splitpath ( filename . c_str ( ) , NULL /* drive */ , NULL /* dir */ , fname , NULL /* ext */ )
module_name = fname ;
2019-02-08 14:40:43 -06:00
# else
2019-02-19 11:02:37 -06:00
char * bn = strdup ( filename . c_str ( ) ) ;
module_name = RTLIL : : escape_id ( bn ) ;
free ( bn ) ;
2019-02-08 14:40:43 -06:00
# endif
}
2019-02-12 14:58:10 -06:00
AigerReader reader ( design , * f , module_name , clk_name , map_filename , wideports ) ;
2019-04-08 18:37:22 -05:00
reader . parse_aiger ( ) ;
2019-02-08 15:17:02 -06:00
}
2019-02-06 14:19:48 -06:00
} AigerFrontend ;
YOSYS_NAMESPACE_END