iverilog/vvp/vpip_to_dec.cc

246 lines
6.7 KiB
C++

/*
* Copyright (c) 2002-2010 Larry Doolittle (larry@doolittle.boa.org)
*
* This source code is free software; you can redistribute it
* and/or modify it in source code form under the terms of the GNU
* General Public License as published by the Free Software
* Foundation; either version 2 of the License, or (at your option)
* any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
*/
# include "config.h"
# include "vpi_priv.h"
# include <stdio.h>
# include <string.h>
# include <limits.h> /* for CHAR_BIT */
# include <stdlib.h>
# include <ctype.h>
#ifdef HAVE_MALLOC_H
# include <malloc.h>
#endif
# include <assert.h>
/* If you are allergic to malloc, you can set a stack memory allocation
* here. Otherwise, malloc() is used for the temporary array, so the
* conversion length is unlimited. */
/* #define MAX_DIGITS 20 */
#if SIZEOF_UNSIGNED_LONG * CHAR_BIT >= 64
#define BDIGITS 9
#define BASE 1000000000
#define BBITS 32
#define BMASK 0xffffffff
#else
#if SIZEOF_UNSIGNED_LONG * CHAR_BIT >= 32
#define BDIGITS 4
#define BASE 10000
#define BBITS 16
#define BMASK 0xffff
#else
#error apparent non-conforming word length
#endif
#endif
#define B_IS0(x) ((x) == 0)
#define B_IS1(x) ((x) == 1)
#define B_ISX(x) ((x) == 2)
#define B_ISZ(x) ((x) == 3)
/* The program works by building a base BASE representation of the number
* in the valv array. BBITS bits of the number can be put in at a time.
* Previous values of each valv element are always less than BASE, the
* input val is less than or equal to 2^BBITS, so (valv[i]<<BBITS)+val
* is guaranteed less than or equal to BASE<<BBITS, which is configured
* less than ULONG_MAX. When this number divided by BASE, to get the amount
* propagated as a "carry" to the next array element, the result is again
* less than or equal to 2^BBITS. BBITS and BASE are configured above
* to depend on the "unsigned long" length of the host, for efficiency.
*/
static inline void shift_in(unsigned long *valv, unsigned int vlen, unsigned long val)
{
unsigned int i;
/* printf("shift in %u\n",val); */
for (i=0; i<vlen; i++) {
val=(valv[i]<<BBITS)+val;
valv[i]=val%BASE;
val=val/BASE;
}
if (val!=0)
fprintf(stderr,"internal error: carry out %lu in " __FILE__ "\n",val);
}
/* Since BASE is a power of ten, conversion of each element of the
* valv array to decimal is easy. sprintf(buf,"%d",v) could be made
* to work, I suppose, but for speed and control I prefer to write
* the steps out longhand.
*/
static inline int write_digits(unsigned long v, char **buf,
unsigned int *nbuf, int zero_suppress)
{
char segment[BDIGITS];
int i;
for (i=BDIGITS-1; i>=0; --i) {
segment[i] = '0' + v%10;
v=v/10;
}
for (i=0; i<BDIGITS; ++i) {
if (!(zero_suppress&=(segment[i]=='0'))) {
*(*buf)++=segment[i]; --(*nbuf);
}
}
return zero_suppress;
}
/* bits[0] is the lsb
* bits[nbits-1] is the msb or sign bit
*/
unsigned vpip_bits_to_dec_str(const unsigned char *bits, unsigned int nbits,
char *buf, unsigned int nbuf, int signed_flag)
{
unsigned int idx, vlen;
unsigned int mbits=nbits; /* number of non-sign bits */
unsigned count_x = 0, count_z = 0;
/* Jump through some hoops so we don't have to malloc/free valv
* on every call, and implement an optional malloc-less version. */
static unsigned long *valv=NULL;
static unsigned int vlen_alloc=0;
unsigned long val=0;
int comp=0;
if (signed_flag) {
if (B_ISZ(bits[nbits-1])) count_z++;
else if (B_ISX(bits[nbits-1])) count_x++;
else if (B_IS1(bits[nbits-1])) comp=1;
--mbits;
}
assert(mbits<(UINT_MAX-92)/28);
vlen = ((mbits*28+92)/93+BDIGITS-1)/BDIGITS;
/* printf("vlen=%d\n",vlen); */
#define ALLOC_MARGIN 4
if (!valv || vlen > vlen_alloc) {
if (valv) free(valv);
valv = (unsigned long*)
calloc( vlen+ALLOC_MARGIN, sizeof (*valv));
if (!valv) {perror("malloc"); return 0; }
vlen_alloc=vlen+ALLOC_MARGIN;
} else {
memset(valv,0,vlen*sizeof(valv[0]));
}
for (idx = 0; idx < mbits; idx += 1) {
/* printf("%c ",bits[mbits-idx-1]); */
if (B_ISZ(bits[mbits-idx-1])) count_z++;
else if (B_ISX(bits[mbits-idx-1])) count_x++;
else if (!comp && B_IS1(bits[mbits-idx-1])) ++val;
else if ( comp && B_IS0(bits[mbits-idx-1])) ++val;
if ((mbits-idx-1)%BBITS==0) {
/* make negative 2's complement, not 1's complement */
if (comp && idx==mbits-1) ++val;
shift_in(valv,vlen,val);
val=0;
} else {
val=val+val;
}
}
if (count_x == nbits) {
buf[0] = 'x';
buf[1] = 0;
} else if (count_x > 0) {
buf[0] = 'X';
buf[1] = 0;
} else if (count_z == nbits) {
buf[0] = 'z';
buf[1] = 0;
} else if (count_z > 0) {
buf[0] = 'Z';
buf[1] = 0;
} else {
int i;
int zero_suppress=1;
if (comp) {
*buf++='-';
nbuf--;
/* printf("-"); */
}
for (i=vlen-1; i>=0; i--) {
zero_suppress = write_digits(valv[i],
&buf,&nbuf,zero_suppress);
/* printf(",%.4u",valv[i]); */
}
/* Awkward special case, since we don't want to
* zero suppress down to nothing at all. The only
* way we can still have zero_suppress on in the
* comp=1 case is if mbits==0, and therefore vlen==0.
* We represent 1'sb1 as "-1". */
if (zero_suppress) *buf++='0'+comp;
/* printf("\n"); */
*buf='\0';
}
/* hold on to the memory, since we expect to be called again. */
/* free(valv); */
return 0;
}
void vpip_dec_str_to_bits(unsigned char*bits, unsigned nbits,
const char*buf, bool signed_flag)
{
/* The str string is the decimal value with the least
significant digit first. This loop creates that string by
reversing the order of the buf string. For example, if the
input is "1234", str gets "4321". */
unsigned slen = strlen(buf);
char*str = new char[slen + 1];
for (unsigned idx = 0 ; idx < slen ; idx += 1) {
if (isdigit(buf[slen-idx-1]))
str[idx] = buf[slen-idx-1];
else
str[idx] = '0';
}
str[slen] = 0;
for (unsigned idx = 0 ; idx < nbits ; idx += 1) {
unsigned val = 0;
switch (str[0]) {
case '1':
case '3':
case '5':
case '7':
case '9':
val = 1;
break;
}
bits[idx] = val;
/* Divide the str string by 2 in decimal. */
char*cp = str;
while (*cp) {
unsigned val = cp[0] - '0';
if ((val&1) && (cp > str))
cp[-1] += 5;
cp[0] = '0' + val/2;
cp += 1;
}
}
delete[]str;
}