Skip to main content
  • Home
  • Development
  • Documentation
  • Donate
  • Operational login
  • Browse the archive

swh logo
SoftwareHeritage
Software
Heritage
Archive
Features
  • Search

  • Downloads

  • Save code now

  • Add forge now

  • Help

https://github.com/TakehideSoh/SAF
18 January 2026, 11:53:51 UTC
  • Code
  • Branches (4)
  • Releases (0)
  • Visits
    • Branches
    • Releases
    • HEAD
    • refs/heads/dev
    • refs/heads/main
    • refs/tags/v1.0-cmsb2023
    • refs/tags/v1.1-CMSB2023
    No releases to show
  • 83aaea6
  • /
  • bdd_minisat_all-1.0.2
  • /
  • main.c
Raw File Download Save again
Take a new snapshot of a software origin

If the archived software origin currently browsed is not synchronized with its upstream version (for instance when new commits have been issued), you can explicitly request Software Heritage to take a new snapshot of it.

Use the form below to proceed. Once a request has been submitted and accepted, it will be processed as soon as possible. You can then check its processing state by visiting this dedicated page.
swh spinner

Processing "take a new snapshot" request ...

To reference or cite the objects present in the Software Heritage archive, permalinks based on SoftWare Hash IDentifiers (SWHIDs) must be used.
Select below a type of object currently browsed in order to display its associated SWHID and permalink.

  • content
  • directory
  • revision
  • snapshot
origin badgecontent badge
swh:1:cnt:b758234b3a729b1d4f2d3cd0638d79d0aa873f5b
origin badgedirectory badge
swh:1:dir:e4b48dfb354d10699e262001c0cd6416edb3f793
origin badgerevision badge
swh:1:rev:d26cc9f94a4f79c046ee0cdd3a127a44f7b443b6
origin badgesnapshot badge
swh:1:snp:d3fcf0e446f52630e867f11dee4c2c828bf46951

This interface enables to generate software citations, provided that the root directory of browsed objects contains a citation.cff or codemeta.json file.
Select below a type of object currently browsed in order to generate citations for them.

  • content
  • directory
  • revision
  • snapshot
Generate software citation in BibTex format (requires biblatex-software package)
Generating citation ...
Generate software citation in BibTex format (requires biblatex-software package)
Generating citation ...
Generate software citation in BibTex format (requires biblatex-software package)
Generating citation ...
Generate software citation in BibTex format (requires biblatex-software package)
Generating citation ...
Tip revision: d26cc9f94a4f79c046ee0cdd3a127a44f7b443b6 authored by TakehideSoh on 23 June 2023, 07:02:26 UTC
Merge pull request #2 from TakehideSoh/dev
Tip revision: d26cc9f
main.c
/**************************************************************************************************
MiniSat -- Copyright (c) 2005, Niklas Sorensson
http://www.cs.chalmers.se/Cs/Research/FormalMethods/MiniSat/

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and
associated documentation files (the "Software"), to deal in the Software without restriction,
including without limitation the rights to use, copy, modify, merge, publish, distribute,
sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all copies or
substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT
NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT
OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
**************************************************************************************************/
// Modified to compile with MS Visual Studio 6.0 by Alan Mishchenko
// Modified to implement bdd-based AllSAT solver on top of MiniSat by Takahisa Toda

#include "solver.h"

#ifdef REDUCTION
#include "bdd_interface.h"
#include "bdd_reduce.h"
#endif

#ifdef GMP
#include <gmp.h>
#endif

#include <stdio.h>
#include <stdlib.h>
#include <time.h>
// #include <unistd.h>
#include <signal.h>
// #include <zlib.h>
// #include <sys/time.h>
// #include <sys/resource.h>

//=================================================================================================
// Helpers:

#ifdef REDUCTION
DdManager *dd_mgr = NULL; //!< BDD/ZDD manager for CUDD
#endif

// Reads an input stream to end-of-file and returns the result as a 'char*' terminated by '\0'
// (dynamic allocation in case 'in' is standard input).
//
char *readFile(FILE *in)
{
    char *data = malloc(65536);
    int cap = 65536;
    int size = 0;

    while (!feof(in))
    {
        if (size == cap)
        {
            cap *= 2;
            data = realloc(data, cap);
        }
        size += fread(&data[size], 1, 65536, in);
    }
    data = realloc(data, size + 1);
    data[size] = '\0';

    return data;
}

// static inline double cpuTime(void) {
//     struct rusage ru;
//     getrusage(RUSAGE_SELF, &ru);
//     return (double)ru.ru_utime.tv_sec + (double)ru.ru_utime.tv_usec / 1000000; }

//=================================================================================================
// DIMACS Parser:

static inline void skipWhitespace(char **in)
{
    while ((**in >= 9 && **in <= 13) || **in == 32)
        (*in)++;
}

static inline void skipLine(char **in)
{
    for (;;)
    {
        if (**in == 0)
            return;
        if (**in == '\n')
        {
            (*in)++;
            return;
        }
        (*in)++;
    }
}

static inline int parseInt(char **in)
{
    int val = 0;
    int _neg = 0;
    skipWhitespace(in);
    if (**in == '-')
        _neg = 1, (*in)++;
    else if (**in == '+')
        (*in)++;
    if (**in < '0' || **in > '9')
        fprintf(stderr, "PARSE ERROR! Unexpected char: %c\n", **in), exit(1);
    while (**in >= '0' && **in <= '9')
        val = val * 10 + (**in - '0'),
        (*in)++;
    return _neg ? -val : val;
}

static void readClause(char **in, solver *s, veci *lits)
{
    int parsed_lit, var;
    veci_resize(lits, 0);
    for (;;)
    {
        parsed_lit = parseInt(in);
        if (parsed_lit == 0)
            break;
        var = abs(parsed_lit) - 1;
        veci_push(lits, (parsed_lit > 0 ? toLit(var) : lit_neg(toLit(var))));
    }
}

static lbool parse_DIMACS_main(char *in, solver *s)
{
    veci lits;
    veci_new(&lits);

    for (;;)
    {
        skipWhitespace(&in);
        if (*in == 0)
            break;
        else if (*in == 'c' || *in == 'p')
            skipLine(&in);
        else
        {
            lit *begin;
            readClause(&in, s, &lits);
            begin = veci_begin(&lits);
            if (!solver_addclause(s, begin, begin + veci_size(&lits)))
            {
                veci_delete(&lits);
                return l_False;
            }
        }
    }
    veci_delete(&lits);
    return solver_simplify(s);
}

// Inserts problem into solver. Returns FALSE upon immediate conflict.
//
static lbool parse_DIMACS(FILE *in, solver *s)
{
    char *text = readFile(in);
    lbool ret = parse_DIMACS_main(text, s);
    free(text);
    return ret;
}

//=================================================================================================

void printStats(stats *stats, unsigned long cpu_time, bool interrupted)
{
    double Time = (double)(cpu_time) / (double)(CLOCKS_PER_SEC);
    printf("restarts          : %12llu\n", stats->starts);
    printf("conflicts         : %12.0f           (%9.0f / sec      )\n", (double)stats->conflicts, (double)stats->conflicts / Time);
    printf("decisions         : %12.0f           (%9.0f / sec      )\n", (double)stats->decisions, (double)stats->decisions / Time);
    printf("propagations      : %12.0f           (%9.0f / sec      )\n", (double)stats->propagations, (double)stats->propagations / Time);
    printf("inspects          : %12.0f           (%9.0f / sec      )\n", (double)stats->inspects, (double)stats->inspects / Time);
    printf("conflict literals : %12.0f           (%9.2f %% deleted  )\n", (double)stats->tot_literals, (double)(stats->max_literals - stats->tot_literals) * 100.0 / (double)stats->max_literals);
    printf("cpu time (solve)  : %12.2f sec\t", Time);
    printf("\n");

    printf("refreshes         : %12llu\n", stats->refreshes);
    printf("|obdd|            : %12llu\n", stats->obddsize);

    printf("cache hits        : %12llu\n", stats->ncachehits);
    printf("cache lookup      : %12llu\n", stats->ncachelookup);

#ifdef CUTSETCACHE
    printf("cache type        : cutset\n");
#else
    printf("cache type        : separator\n");
#endif

#ifdef LAZY
    printf("cache frequency   : lazy\n");
#else
    printf("cache frequency   : original\n");
#endif

#ifdef NONBLOCKING
    printf("minisat_all type  : non-blocking\n");
#if defined(BT)
    printf("backtrack method  : bt\n");
#elif defined(BJ)
    printf("backtrack method  : bj\n");
#elif defined(CBJ)
    printf("backtrack method  : cbj\n");
#else
    printf("backtrack method  : bj+cbj\n");
#endif
#ifdef DLEVEL
    printf("1UIP              : dlevel\n");
#else
    printf("1UIP              : sublevel\n");
#endif
#else
    printf("minisat_all type  : blocking\n");
#endif

#ifdef GMP
    printf("gmp               : enabled\n");
    printf("SAT (full)        : ");
    mpz_out_str(stdout, 10, stats->tot_solutions_gmp);
    if (interrupted)
        printf("+");
    printf("\n");
#else
    printf("gmp               : disabled\n");
    printf("SAT (full)        : %12ju", stats->tot_solutions);
    if (stats->tot_solutions >= INTPTR_MAX || interrupted)
        printf("+");
    printf("\n");
#endif
}

volatile sig_atomic_t eflag = 0;
static void SIGINT_handler(int signum)
{
    eflag = 1;
}

//=================================================================================================

static inline void PRINT_USAGE(char *p)
{
    fprintf(stderr, "Usage:\t%s [options] input-file [output-file]\n", (p));
#ifdef NONBLOCKING
#ifdef REFRESH
    fprintf(stderr, "-n<int>\tmaximum number of obdd nodes: if exceeded, obdd is refreshed\n");
#endif
#endif
}

int main(int argc, char **argv)
{
    solver *s = solver_new();
    lbool st;
    FILE *in;
    FILE *out;
    s->stats.clk = clock();

    char *infile = NULL;
    char *outfile = NULL;
    int lim, span, maxnodes;

    /*** RECEIVE INPUTS ***/
    for (int i = 1; i < argc; i++)
    {
        if (argv[i][0] == '-')
        {
            switch (argv[i][1])
            {
            case 'n':
#ifdef NONBLOCKING
#ifdef REFRESH
                maxnodes = atoi(argv[i] + 2);
                if (maxnodes <= 0)
                {
                    PRINT_USAGE(argv[0]);
                    return 0;
                }
                s->stats.maxnodes = maxnodes;
#endif
#endif
                break;
            case '?':
            case 'h':
            default:
                PRINT_USAGE(argv[0]);
                return 0;
            }
        }
        else
        {
            if (infile == NULL)
                infile = argv[i];
            else if (outfile == NULL)
                outfile = argv[i];
            else
            {
                PRINT_USAGE(argv[0]);
                return 0;
            }
        }
    }
    if (infile == NULL)
    {
        PRINT_USAGE(argv[0]);
        return 0;
    }

    in = fopen(infile, "rb");
    if (in == NULL)
        fprintf(stderr, "ERROR! Could not open file: %s\n", argc == 1 ? "<stdin>" : infile), exit(1);
    if (outfile != NULL)
    {
        out = fopen(outfile, "wb");
        if (out == NULL)
            fprintf(stderr, "ERROR! Could not open file: %s\n", argc == 1 ? "<stdin>" : outfile), exit(1);
#ifdef NONBLOCKING
        else
            s->out = out;
#endif
    }
    else
    {
        out = NULL;
    }

    st = parse_DIMACS(in, s);
    fclose(in);

    if (st == l_False)
    {
        solver_delete(s);
        // printf("Trivial problem\nUNSATISFIABLE\n");
        printf("UNSAT\n");
        exit(20);
    }

    s->verbosity = 0;
    if (signal(SIGINT, SIGINT_handler) == SIG_ERR)
    {
        fprintf(stderr, "ERROR! Cound not set signal");
        exit(1);
    }

    st = solver_solve(s, 0, 0);

    // printf("input             : %s\n", infile);
    // printf("variables         : %12d\n",   s->size);
#ifdef CUTSETCACHE
    // printf("cutwidth          : %12d\n",   s->maxcutwidth);
#else
    // printf("pathwidth         : %12d\n",   s->maxpathwidth);
#endif
    if (eflag == 1)
    {
        printf("\n");
        printf("*** INTERRUPTED ***\n");
        // printStats(&s->stats, clock() - s->stats.clk, true);
        printf("\n");
        printf("*** INTERRUPTED ***\n");
    }
    else
    {
        // printStats(&s->stats, clock() - s->stats.clk, false);
    }

    if (outfile != NULL)
        obdd_decompose(out, s->size, s->root);

#ifdef REDUCTION
    if (s->stats.refreshes == 0)
    { // perform reduction if obdd has not been refreshed.
        bdd_init(s->size, 0);
        clock_t starttime_reduce = clock();
        bddp f = bdd_reduce(s->root);
        clock_t endtime_reduce = clock();
        printf("cpu time (reduce) : %12.2f sec\n", (float)(endtime_reduce - starttime_reduce) / (float)(CLOCKS_PER_SEC));
        printf("|bdd|             : %12ju\n", bdd_size(f));
    }
#endif

    solver_delete(s);
    return 0;
}

back to top

Software Heritage — Copyright (C) 2015–2026, The Software Heritage developers. License: GNU AGPLv3+.
The source code of Software Heritage itself is available on our development forge.
The source code files archived by Software Heritage are available under their own copyright and licenses.
Terms of use: Archive access, API— Content policy— Contact— JavaScript license information— Web API