This source file includes following definitions.
- fprintf
- fprintf
- fprintf
- fprintf
- fprintf
- potion_type
- potion_ptr_type
- potion_qptr_type
- potion_fwd
- potion_str_ptr
- potion_gc_alloc
- potion_gc_calloc
- potion_gc_update
- potion_gc_realloc
- potion_data_alloc
- potion_type_default
#ifndef POTION_H
#define POTION_H
#define POTION_SIG "p\07\10n"
#define POTION_VMID 0x79
#define POTION_X86 0
#define POTION_PPC 1
#define POTION_ARM 2
#include <limits.h>
#include <string.h>
#include <fcntl.h>
#include <assert.h>
#include "config.h"
#if POTION_WIN32
# include <sys/stat.h>
# define DLLEXPORT __declspec(dllexport)
#else
# define DLLEXPORT
#endif
#define _XSTR(s) _STR(s)
#define _STR(s) #s
#if HAVE_ASAN_ATTR
# define ATTRIBUTE_NO_ADDRESS_SAFETY_ANALYSIS __attribute__((no_address_safety_analysis))
#else
# define ATTRIBUTE_NO_ADDRESS_SAFETY_ANALYSIS
#endif
#ifdef DEBUG
# ifndef YY_DEBUG
# define YY_DEBUG
# endif
#endif
typedef unsigned long _PN;
typedef unsigned int PN_SIZE, PNType, PNUniq;
typedef struct Potion_State Potion;
typedef volatile _PN PN;
struct PNObject;
struct PNFwd;
struct PNData;
struct PNString;
struct PNBytes;
struct PNDouble;
struct PNClosure;
struct PNProto;
struct PNTuple;
struct PNWeakRef;
struct PNError;
struct PNCont;
struct PNMemory;
struct PNVtable;
#ifndef __WORDSIZE
# if PN_SIZE_T == 8
# define __WORDSIZE 64
# else
# define __WORDSIZE 32
# endif
#endif
#define PN_TNIL 0x250000
#define PN_TNUMBER (1+PN_TNIL)
#define PN_TBOOLEAN (2+PN_TNIL)
#define PN_TDOUBLE (3+PN_TNIL)
#define PN_TINTEGER (4+PN_TNIL)
#define PN_TSTRING (5+PN_TNIL)
#define PN_TWEAK (6+PN_TNIL)
#define PN_TCLOSURE (7+PN_TNIL)
#define PN_TTUPLE (8+PN_TNIL)
#define PN_TSTATE (9+PN_TNIL)
#define PN_TFILE (10+PN_TNIL)
#define PN_TOBJECT (11+PN_TNIL)
#define PN_TVTABLE (12+PN_TNIL)
#define PN_TSOURCE (13+PN_TNIL)
#define PN_TBYTES (14+PN_TNIL)
#define PN_TPROTO (15+PN_TNIL)
#define PN_TLOBBY (16+PN_TNIL)
#define PN_TTABLE (17+PN_TNIL)
#define PN_TLICK (18+PN_TNIL)
#define PN_TFLEX (19+PN_TNIL)
#define PN_TSTRINGS (20+PN_TNIL)
#define PN_TERROR (21+PN_TNIL)
#define PN_TCONT (22+PN_TNIL)
#define PN_TUSER (23+PN_TNIL)
#define vPN(t) struct PN##t * volatile
#define PN_TYPE(x) potion_type((PN)(x))
#define PN_VTYPE(x) (((struct PNObject *)(x))->vt)
#define PN_TYPE_ID(t) ((t)-PN_TNIL)
#define PN_VTABLE(t) (PN_FLEX_AT(P->vts, PN_TYPE_ID(t)))
#define PN_TYPECHECK(t) (PN_TYPE_ID(t) >= 0 && PN_TYPE_ID(t) < PN_FLEX_SIZE(P->vts))
#define PN_NIL ((PN)0)
#define PN_ZERO ((PN)1)
#define PN_FALSE ((PN)2)
#define PN_TRUE ((PN)6)
#define PN_PRIMITIVE 7
#define PN_REF_MASK ~7
#define PN_NONE ((PN_SIZE)-1)
#define POTION_FWD 0xFFFFFFFE
#define POTION_COPIED 0xFFFFFFFF
#define NIL_NAME "nil"
#define NILKIND_NAME "NilKind"
#define PN_FINTEGER 1
#define PN_FBOOLEAN 2
#define PN_TEST(v) ((PN)(v) != PN_FALSE && (PN)(v) != PN_NIL)
#define PN_BOOL(v) ((v) ? PN_TRUE : PN_FALSE)
#define PN_IS_PTR(v) (!PN_IS_INT(v) && ((PN)(v) & PN_REF_MASK))
#define PN_IS_NIL(v) ((PN)(v) == PN_NIL)
#define PN_IS_BOOL(v) ((PN)(v) & PN_FBOOLEAN)
#define PN_IS_INT(v) ((PN)(v) & PN_FINTEGER)
#define PN_IS_DBL(v) (PN_IS_PTR(v) && ({PNType _t = potion_qptr_type((PN)v); _t == PN_TNUMBER || _t == PN_TDOUBLE;}))
#define PN_IS_NUM(v) (PN_IS_INT(v) || PN_IS_DBL(v))
#define PN_IS_TUPLE(v) (PN_IS_PTR(v) && (potion_ptr_type((PN)v) == PN_TTUPLE))
#define PN_IS_STR(v) (PN_IS_PTR(v) && (potion_ptr_type((PN)v) == PN_TSTRING))
#define PN_IS_TABLE(v) (PN_IS_PTR(v) && (potion_ptr_type((PN)v) == PN_TTABLE))
#define PN_IS_CLOSURE(v) (PN_IS_PTR(v) && (potion_ptr_type((PN)v) == PN_TCLOSURE))
#define PN_IS_PROTO(v) (PN_IS_PTR(v) && (potion_ptr_type((PN)v) == PN_TPROTO))
#define PN_IS_REF(v) (PN_IS_PTR(v) && (potion_ptr_type((PN)v) == PN_TWEAK))
#define PN_IS_METACLASS(v) (((struct PNVtable *)v)->meta == PN_NIL)
#define PN_IS_FFIPTR(p) ((PN_IS_PTR(p) && !(p >= (_PN)P->mem && p <= (_PN)P->mem->birth_hi)) \
|| (!PN_IS_PTR(p) && p > (_PN)P->mem->birth_hi))
#define PN_CHECK_STR(obj) if (!PN_IS_STR(obj)) return potion_type_error_want(P, ""#obj, (PN)obj, "String")
#define PN_CHECK_STRB(obj) if (!PN_IS_STR(obj) || (PN_TYPE(obj) != PN_TBYTES)) return potion_type_error_want(P, ""#obj, (PN)obj, "String or Bytes")
#define PN_CHECK_NUM(obj) if (!PN_IS_NUM(obj)) return potion_type_error_want(P, ""#obj, (PN)obj, "Number")
#define PN_CHECK_INT(obj) if (!PN_IS_INT(obj)) return potion_type_error_want(P, ""#obj, (PN)obj, "Integer")
#define PN_CHECK_DBL(obj) if (!PN_IS_DBL(obj)) return potion_type_error_want(P, ""#obj, (PN)obj, "Double")
#define PN_CHECK_BOOL(obj) if (!PN_IS_BOOL(obj)) return potion_type_error_want(P, ""#obj, (PN)obj, "Bool")
#define PN_CHECK_TUPLE(obj) if (!PN_IS_TUPLE(obj)) return potion_type_error_want(P, ""#obj, (PN)obj, "Tuple")
#define PN_CHECK_CLOSURE(obj) if (!PN_IS_CLOSURE(obj)) return potion_type_error_want(P, ""#obj, (PN)obj, "Closure")
#define PN_CHECK_TYPE(obj,type) if (type != PN_TYPE(obj)) return potion_type_error(P, (PN)obj)
#ifdef DEBUG
#define DBG_CHECK_TYPE(obj,type) PN_CHECK_TYPE(obj,type)
#define DBG_CHECK_NUM(obj) PN_CHECK_NUM(obj)
#define DBG_CHECK_INT(obj) PN_CHECK_INT(obj)
#define DBG_CHECK_DBL(obj) PN_CHECK_DBL(obj)
#define DBG_CHECK_TUPLE(obj) PN_CHECK_TUPLE(obj)
#else
#define DBG_CHECK_TYPE(obj,type)
#define DBG_CHECK_NUM(obj)
#define DBG_CHECK_INT(obj)
#define DBG_CHECK_DBL(obj)
#define DBG_CHECK_TUPLE(obj)
#endif
#define PN_NUM(i) ((PN)((((long)(i))<<1) + PN_FINTEGER))
#define PN_INT(x) ((long)((long)(x))>>1)
#define PN_DBL(num) (PN_IS_INT(num) ? (double)PN_INT(num) : ((struct PNDouble *)num)->value)
typedef _PN (*PN_F)(Potion *, PN, PN, ...);
#define PN_PREC 16
#define PN_RAND() PN_NUM(potion_rand_int())
#define PN_STR(x) potion_str(P, x)
#define PN_STRN(x, l) potion_str2(P, x, l)
#define PN_STRCAT(a, b) potion_strcat(P, (a), (b))
#define PN_STR_PTR(x) potion_str_ptr(x)
#define PN_STR_LEN(x) ((struct PNString *)(x))->len
#define PN_STR_B(x) potion_bytes_string(P, PN_NIL, x)
#define PN_CLOSURE(x) ((struct PNClosure *)(x))
#define PN_CLOSURE_F(x) ((struct PNClosure *)(x))->method
#define PN_PROTO(x) ((struct PNProto *)(x))
#define PN_FUNC(f, s) potion_closure_new(P, (PN_F)f, potion_sig(P, s), 0)
#define PN_DEREF(x) ((struct PNWeakRef *)(x))->data
#define PN_DATA(x) ((struct PNData *)(x))->data
#define PN_TOUCH(x) potion_gc_update(P, (PN)(x))
#define PN_ALIGN(o, x) (((((o) - 1) / (x)) + 1) * (x))
#define PN_OBJECT_HEADER \
PNType vt; \
PNUniq uniq
#define PN_FLEX(N, T) typedef struct { PN_OBJECT_HEADER; PN_SIZE len; PN_SIZE siz; T ptr[]; } N
#define PN_FLEX_AT(N, I) ((PNFlex *)(N))->ptr[I]
#define PN_FLEX_SIZE(N) ((PNFlex *)(N))->len
#if PN_SIZE_T == 4
#define PN_NUMHASH(x) x
#else
#define PN_NUMHASH(x) (PNUniq)((x)>>33^(x)^(x)<<11)
#endif
#define PN_UNIQ(x) (PN_IS_PTR(x) ? ((struct PNObject *)(x))->uniq : PN_NUMHASH(x))
#define AS_STR(x) PN_STR_PTR(potion_send(x, PN_string))
#ifdef DEBUG
#define DBG_t(...) \
if (P->flags & DEBUG_TRACE) fprintf(stderr, __VA_ARGS__)
#define DBG_v(...) \
if (P->flags & DEBUG_VERBOSE) fprintf(stderr, __VA_ARGS__)
#define DBG_vt(...) \
if (P->flags & DEBUG_VERBOSE && P->flags & DEBUG_TRACE) fprintf(stderr, __VA_ARGS__)
#define DBG_vi(...) \
if (P->flags & DEBUG_VERBOSE && P->flags & DEBUG_INSPECT) fprintf(stderr, __VA_ARGS__)
#define DBG_c(...) \
if (P->flags & DEBUG_COMPILE) fprintf(stderr, __VA_ARGS__)
#else
#define DBG_t(...)
#define DBG_v(...)
#define DBG_vt(...)
#define DBG_vi(...)
#define DBG_c(...)
#endif
#define PN_IS_EMPTY(T) (PN_TUPLE_LEN(T) == 0)
#define PN_TUP0() potion_tuple_empty(P)
#define PN_TUP(X) potion_tuple_new(P, X)
#define PN_PUSH(T, X) potion_tuple_push(P, T, (PN)X)
#define PN_SHIFT(T) potion_tuple_shift(P, 0, T)
#define PN_GET(T, X) potion_tuple_find(P, T, X)
#define PN_PUT(T, X) potion_tuple_push_unless(P, T, X)
#define PN_GET_TUPLE(t) ((struct PNTuple *)potion_fwd((PN)t))
#define PN_TUPLE_LEN(t) PN_GET_TUPLE(t)->len
#define PN_TUPLE_AT(t, n) PN_GET_TUPLE(t)->set[n]
#define PN_TUPLE_COUNT(T, I, B) ({ \
struct PNTuple * volatile __t##I = PN_GET_TUPLE(T); \
if (__t##I->len != 0) { \
PN_SIZE I; \
for (I = 0; I < __t##I->len; I++) B \
} \
})
#define PN_TUPLE_EACH(T, I, V, B) ({ \
struct PNTuple * volatile __t##V = PN_GET_TUPLE(T); \
if (__t##V->len != 0) { \
PN_SIZE I; \
for (I = 0; I < __t##V->len; I++) { \
PN V = __t##V->set[I]; \
B \
} \
} \
})
struct PNObject {
PN_OBJECT_HEADER;
PN ivars[];
};
struct PNFwd {
unsigned int fwd;
PN_SIZE siz;
PN ptr;
};
struct PNData {
PN_OBJECT_HEADER;
PN_SIZE siz;
char data[];
};
struct PNString {
PN_OBJECT_HEADER;
PN_SIZE len;
char chars[];
};
struct PNBytes {
PN_OBJECT_HEADER;
PN_SIZE len;
PN_SIZE siz;
char chars[];
};
#define PN_MANTISSA(d, n) d->real[1 + n]
struct PNDouble {
PN_OBJECT_HEADER;
double value;
};
struct PNFile {
PN_OBJECT_HEADER;
int fd;
PN path;
mode_t mode;
};
struct PNClosure {
PN_OBJECT_HEADER;
PN_F method;
PN sig;
int arity;
int minargs;
PN_SIZE extra;
PN name;
PN data[];
};
enum PN_AST {
AST_CODE,
AST_VALUE,
AST_ASSIGN,
AST_NOT,
AST_OR,
AST_AND,
AST_CMP,
AST_EQ,
AST_NEQ,
AST_GT,
AST_GTE,
AST_LT,
AST_LTE,
AST_PIPE,
AST_CARET,
AST_AMP,
AST_WAVY,
AST_BITL,
AST_BITR,
AST_PLUS,
AST_MINUS,
AST_INC,
AST_TIMES,
AST_DIV,
AST_REM,
AST_POW,
AST_MSG,
AST_PATH,
AST_QUERY,
AST_PATHQ,
AST_EXPR,
AST_LIST,
AST_BLOCK,
AST_LICK,
AST_PROTO,
AST_DEBUG
};
struct PNSource {
PN_OBJECT_HEADER;
enum PN_AST part;
struct PNSource * volatile a[3];
struct {
#if PN_SIZE_T != 8
PNType fileno:16;
PNType lineno:16;
#else
PNType fileno:32;
PNType lineno:32;
#endif
} loc;
PN line;
};
struct PNProto {
PN_OBJECT_HEADER;
PN source;
PN sig;
PN stack;
PN paths;
PN locals;
PN upvals;
PN values;
PN protos;
PN debugs;
PN tree;
PN_SIZE pathsize, localsize, upvalsize;
PN asmb;
PN_F jit;
int arity;
PN name;
};
struct PNTuple {
PN_OBJECT_HEADER;
PN_SIZE len;
PN_SIZE alloc;
PN set[];
};
struct PNWeakRef {
PN_OBJECT_HEADER;
PN data;
};
struct PNError {
PN_OBJECT_HEADER;
PN parent;
PN message;
PN line, chr;
PN excerpt;
};
struct PNLick {
PN_OBJECT_HEADER;
PN name;
PN attr;
PN inner;
};
struct PNCont {
PN_OBJECT_HEADER;
PN_SIZE len;
PN stack[];
};
static inline PN potion_fwd(PN);
static inline PNType potion_type(PN obj) {
if (PN_IS_INT(obj)) return PN_TNUMBER;
if (PN_IS_BOOL(obj)) return PN_TBOOLEAN;
if (PN_IS_NIL(obj)) return PN_TNIL;
#if 1
while (1) {
struct PNFwd *o = (struct PNFwd *)obj;
if (o->fwd != POTION_FWD)
return ((struct PNObject *)o)->vt;
obj = o->ptr;
}
#else
return ((struct PNObject *)potion_fwd(obj))->vt;
#endif
}
static inline PNType potion_ptr_type(PN obj) {
while (1) {
struct PNFwd *o = (struct PNFwd *)obj;
if (o->fwd != POTION_FWD)
return ((struct PNObject *)o)->vt;
obj = o->ptr;
}
}
#define PN_QUICK_FWD(t, obj) \
if (((struct PNFwd *)obj)->fwd == POTION_FWD) \
obj = (t)(((struct PNFwd *)obj)->ptr);
static inline PNType potion_qptr_type(PN obj) {
PN_QUICK_FWD(PN, obj);
return ((struct PNObject *)obj)->vt;
}
static inline PN potion_fwd(PN obj) {
while (PN_IS_PTR(obj) && ((struct PNFwd *)obj)->fwd == POTION_FWD)
obj = ((struct PNFwd *)obj)->ptr;
return obj;
}
static inline char *potion_str_ptr(PN s) {
if (((struct PNString *)s)->vt == PN_TSTRING)
return ((struct PNString *)s)->chars;
s = potion_fwd(s);
return ((struct PNBytes *)s)->chars;
}
PN_FLEX(PNFlex, PN);
typedef struct { PN_OBJECT_HEADER; PN_SIZE len; PN_SIZE siz; unsigned char ptr[]; } PNAsm;
#define OP_MAX 50
typedef void (*OP_F)(Potion *P, struct PNProto *, PNAsm * volatile *, ...);
typedef struct {
void (*setup) (Potion *, struct PNProto * volatile, PNAsm * volatile *);
void (*stack) (Potion *, struct PNProto * volatile, PNAsm * volatile *, long);
void (*registers)(Potion *, struct PNProto * volatile, PNAsm * volatile *, long);
void (*local) (Potion *, struct PNProto * volatile, PNAsm * volatile *, long, long);
void (*upvals) (Potion *, struct PNProto * volatile, PNAsm * volatile *, long, long, int);
void (*jmpedit) (Potion *, struct PNProto * volatile, PNAsm * volatile *, unsigned char *, int);
OP_F op[OP_MAX];
void (*finish) (Potion *, struct PNProto * volatile, PNAsm * volatile *);
void (*mcache) (Potion *, struct PNVtable * volatile, PNAsm * volatile *);
void (*ivars) (Potion *, PN, PNAsm * volatile *);
} PNTarget;
typedef enum {
EXEC_VM = 0,
EXEC_JIT = 1,
EXEC_DEBUG = 2,
EXEC_CHECK = 3,
EXEC_COMPILE = 4,
MAX_EXEC = 5
} exec_mode_t;
#define EXEC_BITS 4
typedef enum {
MODE_STD = 1<<EXEC_BITS,
#ifdef P2
MODE_P2 = MODE_STD+1,
MODE_P6 = MODE_STD+2,
#endif
MAX_SYNTAX = (1<<(EXEC_BITS+2))-1
} syntax_mode_t;
typedef enum {
DEBUG_INSPECT = 1<<(EXEC_BITS+2),
DEBUG_VERBOSE = 1<<(EXEC_BITS+3)
#ifdef DEBUG
,
DEBUG_TRACE = 1<<(EXEC_BITS+4),
DEBUG_PARSE = 1<<(EXEC_BITS+5),
DEBUG_PARSE_VERBOSE = 1<<(EXEC_BITS+6),
DEBUG_COMPILE = 1<<(EXEC_BITS+7),
DEBUG_GC = 1<<(EXEC_BITS+8),
DEBUG_JIT = 1<<(EXEC_BITS+9)
#endif
} Potion_Flags;
struct Potion_State {
PN_OBJECT_HEADER;
PNTarget target;
struct PNTable *strings;
PN lobby;
PNFlex * volatile vts;
Potion_Flags flags;
struct PNMemory *mem;
PN call, callset;
int prec;
PN input, source;
int yypos;
PNAsm * volatile pbuf;
PN line;
PN_SIZE fileno;
PN unclosed;
};
struct PNMemory {
volatile void *birth_lo, *birth_hi, *birth_cur;
volatile void **birth_storeptr;
volatile void *old_lo, *old_hi, *old_cur;
volatile int collecting, dirty, pass, majors, minors;
void *cstack;
void *protect;
#ifdef DEBUG
double time;
#endif
};
#define POTION_INIT_STACK(x) \
PN __##x = 0x571FF; void *x = (void *)&__##x
void potion_garbagecollect(Potion *, int, int);
PN_SIZE potion_type_size(Potion *, const struct PNObject *);
unsigned long potion_rand_int();
double potion_rand_double();
static inline void *potion_gc_alloc(Potion *P, PNType vt, int siz) {
struct PNMemory *M = P->mem;
struct PNObject *res = 0;
if (siz < sizeof(struct PNFwd))
siz = sizeof(struct PNFwd);
siz = PN_ALIGN(siz, 8);
assert(M);
assert(M->birth_cur);
assert(M->birth_storeptr);
if (M->dirty || (char *)M->birth_cur + siz >= (char *)M->birth_storeptr - 2)
potion_garbagecollect(P, siz + 4 * sizeof(double), 0);
res = (struct PNObject *)M->birth_cur;
res->vt = vt;
res->uniq = (PNUniq)potion_rand_int();
M->birth_cur = (char *)res + siz;
return (void *)res;
}
static inline void *potion_gc_calloc(Potion *P, PNType vt, int siz) {
return potion_gc_alloc(P, vt, siz);
}
static inline void potion_gc_update(Potion *P, PN x) {
struct PNMemory *M = P->mem;
if ((x > (PN)M->birth_lo && x < (PN)M->birth_hi && (x < (PN)M || x >= (PN)M->protect)) ||
x == (PN)M->birth_storeptr[1] ||
x == (PN)M->birth_storeptr[2] ||
x == (PN)M->birth_storeptr[3])
return;
*(M->birth_storeptr--) = (void *)x;
if ((void **)M->birth_storeptr - 4 <= (void **)M->birth_cur)
potion_garbagecollect(P, POTION_PAGESIZE, 0);
}
static inline void *potion_gc_realloc(Potion *P, PNType vt, struct PNObject * volatile obj, PN_SIZE sz) {
void *dst = 0;
PN_SIZE oldsz = 0;
if (obj != NULL) {
oldsz = potion_type_size(P, (const struct PNObject *)obj);
if (oldsz >= sz)
return (void *)obj;
}
dst = potion_gc_alloc(P, vt, sz);
if (obj != NULL) {
memcpy(dst, (void *)obj, oldsz);
((struct PNFwd *)obj)->fwd = POTION_FWD;
((struct PNFwd *)obj)->siz = oldsz;
((struct PNFwd *)obj)->ptr = (PN)dst;
potion_gc_update(P, (PN)obj);
}
return dst;
}
static inline struct PNData *potion_data_alloc(Potion *P, int siz) {
struct PNData *data = potion_gc_alloc(P, PN_TUSER, sizeof(struct PNData) + siz);
data->siz = siz;
return data;
}
#define POTION_OK 0
#define POTION_NO_MEM 8910
#define potion_send(RCV, MSG, ARGS...) ({ \
PN r = (PN)(RCV); \
PN c = potion_bind(P, r, (MSG)); \
if (PN_IS_CLOSURE(c)) \
c = ((struct PNClosure *)c)->method(P, c, r, ##ARGS); \
c; \
})
#define potion_method(RCV, MSG, FN, SIG) \
potion_send(RCV, PN_def, potion_str(P, MSG), PN_FUNC(FN, SIG))
#define potion_class_method(RCV, MSG, FN, SIG) \
potion_send(((struct PNVtable *)(RCV))->meta, PN_def, potion_str(P, MSG), PN_FUNC(FN, SIG))
extern PN PN_allocate, PN_break, PN_call, PN_class, PN_compile,
PN_continue, PN_def, PN_delegated, PN_else, PN_elsif, PN_if,
PN_lookup, PN_loop, PN_print, PN_return, PN_self, PN_string,
PN_while;
extern PN PN_add, PN_sub, PN_mult, PN_div, PN_rem, PN_bitn, PN_bitl, PN_bitr;
extern PN PN_cmp, PN_number, PN_name, PN_length, PN_size, PN_STR0;
extern PN PN_extern, PN_integer;
extern PN pn_filenames;
static inline PN potion_type_default(char type) {
return type == 'N' ? PN_ZERO
: type == 'I' ? PN_ZERO
: type == 'S' ? PN_STR0
: PN_NIL;
}
Potion *potion_create(void *);
void potion_destroy(Potion *);
PN potion_error(Potion *, PN, long, long, PN);
void potion_fatal(char *);
void potion_allocation_error(void);
PN potion_io_error(Potion *, const char *);
PN potion_type_error(Potion *, PN);
PN potion_type_error_want(Potion *, const char *, PN, const char *);
void potion_syntax_error(Potion *, struct PNSource *, const char *, ...)
__attribute__ ((format (printf, 3, 4)));
PNType potion_kind_of(PN);
void potion_p(Potion *, PN);
PN potion_str(Potion *, const char *);
PN potion_str2(Potion *, char *, size_t);
PN potion_strcat(Potion *P, char *str, char *str2);
PN potion_str_add(Potion *P, PN, PN, PN);
PN potion_str_format(Potion *, const char *, ...)
__attribute__ ((format (printf, 2, 3)));
PN potion_byte_str(Potion *, const char *);
PN potion_byte_str2(Potion *, const char *, size_t len);
PN potion_bytes(Potion *, size_t);
PN potion_bytes_string(Potion *, PN, PN);
PN_SIZE pn_printf(Potion *, PN, const char *, ...)
__attribute__ ((format (printf, 3, 4)));
void potion_bytes_obj_string(Potion *, PN, PN);
PN potion_bytes_append(Potion *, PN, PN, PN);
void potion_release(Potion *, PN);
PN potion_def_method(Potion *P, PN, PN, PN, PN);
PN potion_type_new(Potion *, PNType, PN);
PN potion_type_new2(Potion *, PNType, PN, PN);
void potion_type_call_is(PN, PN);
void potion_type_callset_is(PN, PN);
void potion_type_constructor_is(PN, PN);
char potion_type_char(PNType);
PN potion_class(Potion *, PN, PN, PN);
PN potion_ivars(Potion *, PN, PN, PN);
PN potion_obj_get_call(Potion *, PN);
PN potion_obj_get_callset(Potion *, PN);
PN potion_obj_get(Potion *, PN, PN, PN);
PN potion_obj_set(Potion *, PN, PN, PN, PN);
PN potion_object_new(Potion *, PN, PN);
PN potion_delegated(Potion *, PN, PN);
PN potion_call(Potion *, PN, PN_SIZE, PN * volatile);
PN potion_lookup(Potion *, PN, PN, PN);
PN potion_bind(Potion *, PN, PN);
PN potion_message(Potion *, PN, PN);
PN potion_closure_new(Potion *, PN_F, PN, PN_SIZE);
PN potion_callcc(Potion *, PN, PN);
PN potion_ref(Potion *, PN);
PN potion_sig(Potion *, char *);
int potion_sig_find(Potion *, PN, PN);
PN potion_double(Potion *, double);
PN potion_strtod(Potion *, char *, int);
PN potion_num_pow(Potion *, PN, PN, PN);
PN potion_num_rand(Potion *, PN, PN);
PN potion_srand(Potion *, PN, PN, PN);
PN potion_rand(Potion *, PN, PN);
PN potion_sig_at(Potion *, PN, int);
PN potion_sig_name_at(Potion *, PN, int);
int potion_sig_arity(Potion *, PN);
int potion_sig_minargs(Potion *, PN);
PN potion_closure_arity(Potion *, PN, PN);
PN potion_closure_minargs(Potion *, PN, PN);
void potion_define_global(Potion *, PN, PN);
PN potion_obj_add(Potion *, PN, PN);
PN potion_obj_sub(Potion *, PN, PN);
PN potion_obj_mult(Potion *, PN, PN);
PN potion_obj_div(Potion *, PN, PN);
PN potion_obj_rem(Potion *, PN, PN);
PN potion_obj_bitn(Potion *, PN);
PN potion_obj_bitl(Potion *, PN, PN);
PN potion_obj_bitr(Potion *, PN, PN);
PN potion_any_cmp(Potion *, PN, PN, PN);
PN potion_tuple_empty(Potion *);
PN potion_tuple_with_size(Potion *, unsigned long);
PN potion_tuple_new(Potion *, PN);
PN potion_tuple_push(Potion *, PN, PN);
PN_SIZE potion_tuple_push_unless(Potion *, PN, PN);
PN_SIZE potion_tuple_find(Potion *, PN, PN);
PN potion_tuple_at(Potion *, PN, PN, PN);
PN potion_tuple_shift(Potion *, PN, PN);
PN potion_tuple_bsearch(Potion *, PN, PN, PN);
PN potion_tuple_ins_sort(Potion *, PN, PN, PN);
PN potion_table_empty(Potion *);
PN potion_table_put(Potion *, PN, PN, PN, PN);
PN potion_table_set(Potion *, PN, PN, PN);
PN potion_table_at(Potion *, PN, PN, PN);
PN potion_lick(Potion *, PN, PN, PN);
PN potion_source_compile(Potion *, PN, PN, PN, PN);
PN potion_source_load(Potion *, PN, PN);
PN potion_source_dump(Potion *, PN, PN, PN, PN);
PN potion_source_dumpbc(Potion *, PN, PN, PN);
PN potion_greg_parse(Potion *, PN);
PN potion_sig_string(Potion *, PN, PN);
PN potion_filename_find(Potion *, PN);
PN potion_filename_push(Potion *, PN);
Potion *potion_gc_boot(void *);
void potion_lobby_init(Potion *);
void potion_object_init(Potion *);
void potion_error_init(Potion *);
void potion_primitive_init(Potion *);
void potion_num_init(Potion *);
void potion_str_hash_init(Potion *);
void potion_str_init(Potion *);
void potion_table_init(Potion *);
void potion_source_init(Potion *);
void potion_lick_init(Potion *);
void potion_compiler_init(Potion *);
void potion_vm_init(Potion *);
void potion_cont_init(Potion *);
#ifndef SANDBOX
void potion_file_init(Potion *);
void potion_loader_init(Potion *);
void potion_loader_add(Potion *, PN path);
PN potion_load(Potion *, PN, PN, PN);
char *potion_find_file(Potion *, char *str, PN_SIZE str_len);
#endif
#if defined(STATIC) || defined(SANDBOX)
void Potion_Init_readline(Potion *);
void Potion_Init_buffile(Potion *);
void Potion_Init_aio(Potion *);
#endif
void potion_dump_stack(Potion *);
PN potion_num_string(Potion *, PN, PN);
PN potion_gc_reserved(Potion *, PN, PN);
PN potion_gc_actual(Potion *, PN, PN);
PN potion_gc_fixed(Potion *, PN, PN);
PN potion_parse(Potion *, PN, char*);
PN potion_vm_proto(Potion *, PN, PN, ...);
PN potion_vm_class(Potion *, PN, PN);
PN potion_vm(Potion *, PN, PN, PN, PN_SIZE, PN * volatile);
PN potion_eval(Potion *, PN);
PN potion_run(Potion *, PN, int);
PN_F potion_jit_proto(Potion *, PN);
PN potion_class_find(Potion *, PN);
PNType potion_class_type(Potion *, PN);
#endif