Espresso.mod

Espresso功能强大。处理16 in 40 out时不仅速度很快,而且只用了494x359的矩阵。集合则多到了用set family的程度,所占空间我没数,想必不会太多,因为494x359它就用稀疏矩阵来存了,row和column都是排序的双链表。那个时代内存好贵。

一个简单的例子: 输入:
i 2 o 3
00 000
01 001
10 011
11 111
计算2**n - 1的真值表。输入为2位ab,输出为3位xyz. Espresso处理后:
11 100
-1 001
1- 011
11代表a&b, -1代表b,1-代表a。100, 001和011中只有100的x为1,对应的是11,即x=a&b. 然后y=a, c等于b|a. 真值表正确。

gcc -w *.c即可编译,a.out或a.exe filename处理。*.txt是测试文件。博客园下载 101KB 含a.exe. -w 是忽略所有警告。

int main(int argc, char* argv[]) {
    puts("A simplified version of Espresso V2.3 (Released by UC Berkeley on 01/31/1988)");
    if (argc != 2) fatal("Usage: %s file", argv[0]);
    // F: on-set, R: off-set, D: don't case-set
    PLA_t*    p = pla_load(argv[1]);    // PLA: Programmable Logic Array
    pset_family    F = sf_save(p->F);    // sf: set_family
    p->F = espresso(p->F, p->D, p->R);    // returns a set_family*
    if (verify(p->F, F, p->D)) fatal("Verification failed");
    sf_free(F);
    pla_print(p); pla_free(p);
    cube_cleanup(); sf_cleanup();
    extern int    max_row, max_col, irred;
    printf("matrix: max_row=%d, max_col=%d irred=%d\n", max_row, max_col, irred);
}

main很好懂,如果把espresso看作黑盒子的话。不支持复杂格式的pla_load一点都不mess:

static PLA_t* pla_load(const char* fname) {
    FILE*    fp = fopen(fname, "rt");
    if (!fp) fatal("Unable to open file");
    char    s[256], _[64];
    int     n_i, n_o;
    fgets(s, sizeof(s), fp); sscanf(s, "%s%d%s%d", _, &n_i, _, &n_o);
    printf("%d inputs and %d outputs\n", n_i, n_o);
    PLA_t*    p = (PLA_t*)calloc(1, sizeof(PLA_t));
    cube.num_vars = (cube.num_binary_vars = n_i) + 1;
    cube.part_size = ALLOC(int, n_i + 1); cube.part_size[n_i] = n_o;
    cube_setup();
    p->F = new_cover(10); p->D = new_cover(10);
    while (fgets(s, sizeof(s), fp)) {
        const char*    ss = s;
        pcube    cf = cube.temp[0], cd = cube.temp[1];
        int        savef = 0, saved = 0;
        set_clear(cf, cube.size);
        for(int var = 0; var < n_i; var++) {
            switch(*ss++) {
            case '0': set_insert(cf, var * 2); break;
            case '1': case '-': set_insert(cf, var * 2 + 1); break;
            default: fatal("Error: %s", s);
            }
        }
        while (*ss++ == ' '); --ss;
        set_copy(cd, cf);
        for(int i = cube.first_part[n_i]; i <= cube.last_part[n_i]; i++) {
            switch(*ss++) {
            case '0': break;
            case '1': set_insert(cf, i); savef = TRUE; break;
            case '-': set_insert(cd, i); saved = TRUE;break;
            default: fatal("Error: %s", s);
            }
        }
        if (savef) p->F = sf_addset(p->F, cf);
        if (saved) p->D = sf_addset(p->D, cd);
    }
    fclose(fp);
    p->R = complement(cube2list(p->F, p->D));
    pla_print_cost(p);
    return p;
}

然后我实在整不动了。我倒是知道数一个32-bit-int里的1的个数的方法,一本叫"Hacker's Delight"的书里讲过,ffmpeg的源码里也有类似的。可他们是怎么想出来的?Espresso好像也不能解决这个问题: 它倒是能给出逻辑表达式,但一串not, and, or,不是shift, xor等。数1的话,把int分成4个byte来数也不慢,事实上Espresso有个bit_count[256].

上一篇:Oracle 其他函数


下一篇:Kylin系列7-Cube 构建优化