@ -83,7 +83,7 @@ uint_fast64_t NonDeterministicSparseTransitionParser::firstPass(char* buf, uint_
maxnode = 0 ;
maxnode = 0 ;
while ( buf [ 0 ] ! = ' \0 ' ) {
while ( buf [ 0 ] ! = ' \0 ' ) {
/*
/*
* Read source node and choice .
* Read source node .
* Check if current source node is larger than current maximum node id .
* Check if current source node is larger than current maximum node id .
* Increase number of choices .
* Increase number of choices .
* Check if we have skipped any source node , i . e . if any node has no
* Check if we have skipped any source node , i . e . if any node has no
@ -97,6 +97,8 @@ uint_fast64_t NonDeterministicSparseTransitionParser::firstPass(char* buf, uint_
nonzero + = source - lastsource - 1 ;
nonzero + = source - lastsource - 1 ;
parsed_nonzero + = source - lastsource - 1 ;
parsed_nonzero + = source - lastsource - 1 ;
}
}
lastsource = source ;
buf = trimWhitespaces ( buf ) ; // Skip to name of choice
buf + = strcspn ( buf , " \t \n \r " ) ; // Skip name of choice.
buf + = strcspn ( buf , " \t \n \r " ) ; // Skip name of choice.
/*
/*
@ -153,85 +155,134 @@ NonDeterministicSparseTransitionParser::NonDeterministicSparseTransitionParser(s
: matrix ( nullptr )
: matrix ( nullptr )
{
{
/*
/*
* e nforce locale where decimal point is ' . '
* E nforce locale where decimal point is ' . ' .
*/
*/
setlocale ( LC_NUMERIC , " C " ) ;
setlocale ( LC_NUMERIC , " C " ) ;
/*
/*
* open file
* Open file .
*/
*/
MappedFile file ( filename . c_str ( ) ) ;
MappedFile file ( filename . c_str ( ) ) ;
char * buf = file . data ;
char * buf = file . data ;
/*
/*
* p erform first pass , i . e . count entries that are not zero and not on the diagonal
* P erform first pass , i . e . obtain number of columns , rows and non - zero elements .
*/
*/
uint_fast64_t maxnode , max choices;
uint_fast64_t non_ zero = this - > firstPass ( file . data , maxnode , maxchoices ) ;
uint_fast64_t maxnode , choices ;
uint_fast64_t nonzero = this - > firstPass ( file . data , choices , maxnode ) ;
/*
/*
* i f first pass returned zero , the file format was wrong
* I f first pass returned zero , the file format was wrong .
*/
*/
if ( non_ zero = = 0 )
if ( nonzero = = 0 )
{
{
LOG4CPLUS_ERROR ( logger , " Error while parsing " < < filename < < " : erroneous file format. " ) ;
LOG4CPLUS_ERROR ( logger , " Error while parsing " < < filename < < " : erroneous file format. " ) ;
throw storm : : exceptions : : WrongFileFormatException ( ) ;
throw storm : : exceptions : : WrongFileFormatException ( ) ;
}
}
/*
/*
* p erform second pass
* P erform second pass .
*
*
* f rom here on , we already know that the file header is correct
* F rom here on , we already know that the file header is correct .
*/
*/
/*
/*
* r ead file header , extract number of states
* R ead file header , ignore values within .
*/
*/
buf + = 7 ; // skip "STATES "
buf = strchr ( buf , ' \n ' ) + 1 ; // skip format hint
buf + = 7 ; // skip "STATES "
checked_strtol ( buf , & buf ) ;
checked_strtol ( buf , & buf ) ;
buf = trimWhitespaces ( buf ) ;
buf = trimWhitespaces ( buf ) ;
buf + = 12 ; // skip "TRANSITIONS "
buf + = 12 ; // skip "TRANSITIONS "
checked_strtol ( buf , & buf ) ;
checked_strtol ( buf , & buf ) ;
/*
/*
* Creating matrix
* Memory for diagonal elements is automatically allocated , hence only the number of non - diagonal
* non - zero elements has to be specified ( which is non_zero , computed by make_first_pass )
* Create and initialize matrix .
* The matrix should have as many columns as we have nodes and as many rows as we have choices .
* Those two values , as well as the number of nonzero elements , was been calculated in the first run .
*/
*/
LOG4CPLUS_INFO ( logger , " Attempting to create matrix of size " < < ( maxnode + 1 ) < < " x " < < ( maxnode + 1 ) < < " . " ) ;
this - > matrix = std : : shared_ptr < storm : : storage : : SparseMatrix < double > > ( new storm : : storage : : SparseMatrix < double > ( maxnode + 1 ) ) ;
if ( this - > matrix = = NULL )
{
LOG4CPLUS_ERROR ( logger , " Could not create matrix of size " < < ( maxnode + 1 ) < < " x " < < ( maxnode + 1 ) < < " . " ) ;
LOG4CPLUS_INFO ( logger , " Attempting to create matrix of size " < < choices < < " x " < < ( maxnode + 1 ) < < " with " < < nonzero < < " entries. " ) ;
this - > matrix = std : : shared_ptr < storm : : storage : : SparseMatrix < double > > ( new storm : : storage : : SparseMatrix < double > ( choices , maxnode + 1 ) ) ;
if ( this - > matrix = = nullptr ) {
LOG4CPLUS_ERROR ( logger , " Could not create matrix of size " < < choices < < " x " < < ( maxnode + 1 ) < < " . " ) ;
throw std : : bad_alloc ( ) ;
throw std : : bad_alloc ( ) ;
}
}
// TODO: put stuff in matrix / matrices.
//this->matrix->initialize(*non_zero);
this - > matrix - > initialize ( nonzero ) ;
uint_fast64_t row , col , ndchoice ;
/*
* Parse file content .
*/
uint_fast64_t source , target , lastsource = 0 ;
uint_fast64_t curRow = 0 ;
std : : string choice ;
double val ;
double val ;
bool fixDeadlocks = storm : : settings : : instance ( ) - > isSet ( " fix-deadlocks " ) ;
bool hadDeadlocks = false ;
/*
/*
* read all transitions from file
* R ead all transitions from file .
*/
*/
while ( buf [ 0 ] ! = ' \0 ' )
while ( buf [ 0 ] ! = ' \0 ' )
{
{
/*
/*
* read row , col and value .
* Read source node and choice name .
*/
source = checked_strtol ( buf , & buf ) ;
buf = trimWhitespaces ( buf ) ; // Skip to name of choice
choice = std : : string ( buf , strcspn ( buf , " \t \n \r " ) ) ;
/*
* Check if we have skipped any source node , i . e . if any node has no
* outgoing transitions . If so , insert a self - loop .
*/
for ( uint_fast64_t node = lastsource + 1 ; node < source ; node + + ) {
hadDeadlocks = true ;
if ( fixDeadlocks ) {
// TODO: add (curRow <-> (node,"")) to rowMapping
this - > matrix - > addNextValue ( curRow , node , 1 ) ;
curRow + + ;
LOG4CPLUS_WARN ( logger , " Warning while parsing " < < filename < < " : node " < < node < < " has no outgoing transitions. A self-loop was inserted. " ) ;
} else {
LOG4CPLUS_ERROR ( logger , " Error while parsing " < < filename < < " : node " < < node < < " has no outgoing transitions. " ) ;
}
}
lastsource = source ;
// TODO: add (curRow <-> (node,choice)) to rowMapping
/*
* Skip name of choice .
*/
*/
row = checked_strtol ( buf , & buf ) ;
ndchoice = checked_strtol ( buf , & buf ) ;
col = checked_strtol ( buf , & buf ) ;
val = strtod ( buf , & buf ) ;
buf + = strcspn ( buf , " \t \n \r " ) ;
//this->matrix->addNextValue(row,col,val);
/*
* Read all targets for this choice .
*/
buf = trimWhitespaces ( buf ) ;
buf = trimWhitespaces ( buf ) ;
while ( buf [ 0 ] = = ' * ' ) {
buf + + ;
/*
* Read target node and transition value .
* Put it into the matrix .
*/
target = checked_strtol ( buf , & buf ) ;
val = checked_strtod ( buf , & buf ) ;
this - > matrix - > addNextValue ( curRow , target , val ) ;
/*
* Proceed to beginning of next line in file and next row in matrix .
*/
buf = trimWhitespaces ( buf ) ;
}
curRow + + ;
}
}
if ( ! fixDeadlocks & & hadDeadlocks ) throw storm : : exceptions : : WrongFileFormatException ( ) < < " Some of the nodes had deadlocks. You can use --fix-deadlocks to insert self-loops on the fly. " ;
/*
/*
* clean up
* Finalize matrix .
*/
*/
//this->matrix->finalize();
this - > matrix - > finalize ( ) ;
}
}
} //namespace parser
} //namespace parser