urbit/noun/retrieve.c

1730 lines
33 KiB
C
Raw Normal View History

2014-09-11 04:01:32 +04:00
/* g/r.c
2014-09-05 23:55:16 +04:00
**
*/
#include "all.h"
#include <murmur3.h>
2014-09-05 23:55:16 +04:00
/* _frag_word(): fast fragment/branch prediction for top word.
*/
2014-09-06 00:13:24 +04:00
static u3_weak
_frag_word(c3_w a_w, u3_noun b)
2014-09-05 23:55:16 +04:00
{
c3_assert(0 != a_w);
{
2014-11-18 00:56:51 +03:00
c3_w dep_w = u3x_dep(a_w);
2014-09-05 23:55:16 +04:00
while ( dep_w ) {
2014-11-06 03:20:01 +03:00
if ( c3n == u3a_is_cell(b) ) {
2014-11-06 02:36:30 +03:00
return u3_none;
2014-09-05 23:55:16 +04:00
}
else {
2014-11-06 03:20:01 +03:00
u3a_cell* b_u = u3a_to_ptr(b);
2014-09-05 23:55:16 +04:00
2014-09-06 00:13:24 +04:00
b = *(((u3_noun*)&(b_u->hed)) + (1 & (a_w >> (dep_w - 1))));
2014-09-05 23:55:16 +04:00
dep_w--;
}
}
return b;
}
}
/* _frag_deep(): fast fragment/branch for deep words.
*/
2014-09-06 00:13:24 +04:00
static u3_weak
_frag_deep(c3_w a_w, u3_noun b)
2014-09-05 23:55:16 +04:00
{
c3_w dep_w = 32;
while ( dep_w ) {
2014-11-06 03:20:01 +03:00
if ( c3n == u3a_is_cell(b) ) {
2014-11-06 02:36:30 +03:00
return u3_none;
2014-09-05 23:55:16 +04:00
}
else {
2014-11-06 03:20:01 +03:00
u3a_cell* b_u = u3a_to_ptr(b);
2014-09-05 23:55:16 +04:00
2014-09-06 00:13:24 +04:00
b = *(((u3_noun*)&(b_u->hed)) + (1 & (a_w >> (dep_w - 1))));
2014-09-05 23:55:16 +04:00
dep_w--;
}
}
return b;
}
2014-11-06 03:20:01 +03:00
/* u3r_at():
2014-09-05 23:55:16 +04:00
**
2014-11-06 02:36:30 +03:00
** Return fragment (a) of (b), or u3_none if not applicable.
2014-09-05 23:55:16 +04:00
*/
2014-09-06 00:13:24 +04:00
u3_weak
2014-12-03 00:53:35 +03:00
u3r_at(u3_atom a, u3_noun b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 02:36:30 +03:00
c3_assert(u3_none != a);
c3_assert(u3_none != b);
2014-09-05 23:55:16 +04:00
2014-12-03 00:53:35 +03:00
u3t_on(far_o);
2014-09-05 23:55:16 +04:00
if ( 0 == a ) {
2014-12-03 00:53:35 +03:00
u3t_off(far_o);
2014-11-06 02:36:30 +03:00
return u3_none;
2014-09-05 23:55:16 +04:00
}
2014-11-06 03:20:01 +03:00
if ( _(u3a_is_cat(a)) ) {
2014-12-03 00:53:35 +03:00
u3t_off(far_o);
2014-09-05 23:55:16 +04:00
return _frag_word(a, b);
}
else {
2014-11-06 03:20:01 +03:00
if ( !_(u3a_is_pug(a)) ) {
2014-12-03 00:53:35 +03:00
u3t_off(far_o);
2014-11-06 02:36:30 +03:00
return u3_none;
2014-09-05 23:55:16 +04:00
}
else {
2014-11-06 03:20:01 +03:00
u3a_atom* a_u = u3a_to_ptr(a);
2014-09-05 23:55:16 +04:00
c3_w len_w = a_u->len_w;
b = _frag_word(a_u->buf_w[len_w - 1], b);
len_w -= 1;
2018-11-14 03:22:49 +03:00
if ( u3_none == b ) {
u3t_off(far_o);
return b;
}
2014-09-05 23:55:16 +04:00
while ( len_w ) {
2014-09-15 07:27:48 +04:00
b = _frag_deep(a_u->buf_w[len_w - 1], b);
2014-09-05 23:55:16 +04:00
2014-11-06 02:36:30 +03:00
if ( u3_none == b ) {
2014-12-03 00:53:35 +03:00
u3t_off(far_o);
2014-09-05 23:55:16 +04:00
return b;
} else {
len_w--;
}
}
2014-12-03 00:53:35 +03:00
u3t_off(far_o);
2014-09-05 23:55:16 +04:00
return b;
}
}
}
2014-11-06 03:20:01 +03:00
/* u3r_mean():
2014-09-05 23:55:16 +04:00
**
** Attempt to deconstruct `a` by axis, noun pairs; 0 terminates.
** Axes must be sorted in tree order.
*/
struct _mean_pair {
c3_w axe_w;
2014-09-06 00:13:24 +04:00
u3_noun* som;
2014-09-05 23:55:16 +04:00
};
static c3_w
_mean_cut(c3_w len_w,
struct _mean_pair* prs_m)
{
c3_w i_w, cut_t, cut_w;
2014-11-05 04:09:17 +03:00
cut_t = 0;
2014-09-05 23:55:16 +04:00
cut_w = 0;
for ( i_w = 0; i_w < len_w; i_w++ ) {
c3_w axe_w = prs_m[i_w].axe_w;
2014-11-18 00:56:51 +03:00
if ( (cut_t == 0) && (3 == u3x_cap(axe_w)) ) {
2014-11-05 04:09:17 +03:00
cut_t = 1;
2014-09-05 23:55:16 +04:00
cut_w = i_w;
}
2014-11-18 00:56:51 +03:00
prs_m[i_w].axe_w = u3x_mas(axe_w);
2014-09-05 23:55:16 +04:00
}
return cut_t ? cut_w : i_w;
}
2014-11-06 06:10:22 +03:00
static c3_o
2014-09-06 00:13:24 +04:00
_mean_extract(u3_noun som,
2014-09-05 23:55:16 +04:00
c3_w len_w,
struct _mean_pair* prs_m)
{
if ( len_w == 0 ) {
2014-11-05 04:18:47 +03:00
return c3y;
2014-09-05 23:55:16 +04:00
}
else if ( (len_w == 1) && (1 == prs_m[0].axe_w) ) {
*prs_m->som = som;
2014-11-05 04:18:47 +03:00
return c3y;
2014-09-05 23:55:16 +04:00
}
else {
2014-11-06 03:20:01 +03:00
if ( c3n == u3a_is_cell(som) ) {
2014-11-05 04:18:47 +03:00
return c3n;
2014-09-05 23:55:16 +04:00
} else {
c3_w cut_w = _mean_cut(len_w, prs_m);
2014-11-05 04:18:47 +03:00
return c3a
2014-11-06 03:20:01 +03:00
(_mean_extract(u3a_h(som), cut_w, prs_m),
_mean_extract(u3a_t(som), (len_w - cut_w), (prs_m + cut_w)));
2014-09-05 23:55:16 +04:00
}
}
}
__attribute__((no_sanitize("address")))
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_mean(u3_noun som,
2014-09-05 23:55:16 +04:00
...)
{
va_list ap;
c3_w len_w;
struct _mean_pair* prs_m;
2014-11-06 02:36:30 +03:00
c3_assert(u3_none != som);
2014-09-05 23:55:16 +04:00
/* Count.
*/
len_w = 0;
{
va_start(ap, som);
while ( 1 ) {
if ( 0 == va_arg(ap, c3_w) ) {
break;
}
2014-09-06 00:13:24 +04:00
va_arg(ap, u3_noun*);
2014-09-05 23:55:16 +04:00
len_w++;
}
va_end(ap);
}
c3_assert( 0 != len_w );
2014-09-05 23:55:16 +04:00
prs_m = alloca(len_w * sizeof(struct _mean_pair));
/* Install.
*/
{
c3_w i_w;
va_start(ap, som);
for ( i_w = 0; i_w < len_w; i_w++ ) {
prs_m[i_w].axe_w = va_arg(ap, c3_w);
2014-09-06 00:13:24 +04:00
prs_m[i_w].som = va_arg(ap, u3_noun*);
2014-09-05 23:55:16 +04:00
}
va_end(ap);
}
/* Extract.
*/
return _mean_extract(som, len_w, prs_m);
}
2015-06-08 22:48:35 +03:00
/* _sang_one(): unify but leak old.
*/
2015-06-08 22:48:35 +03:00
static void
_sang_one(u3_noun* a, u3_noun* b)
{
if ( *a == *b ) {
return;
}
else {
c3_o asr_o = u3a_is_senior(u3R, *a);
c3_o bsr_o = u3a_is_senior(u3R, *b);
if ( _(asr_o) && _(bsr_o) ) {
// You shouldn't have let this happen. We don't want to
// descend down to a lower road and free there, because
// synchronization - though this could be revisited under
// certain circumstances.
//
return;
}
if ( _(asr_o) && !_(bsr_o) ){
// u3z(*b);
*b = *a;
}
if ( _(bsr_o) && !_(asr_o) ) {
// u3z(*a);
*a = *b;
}
if ( u3a_is_north(u3R) ) {
if ( *a <= *b ) {
u3k(*a);
// u3z(*b);
*b = *a;
} else {
u3k(*b);
// u3z(*a);
*a = *b;
}
}
else {
if ( *a >= *b ) {
u3k(*a);
// u3z(*b);
*b = *a;
} else {
u3k(*b);
// u3z(*a);
*a = *b;
}
}
}
}
#define SONG_NONE 0
#define SONG_HEAD 1
#define SONG_TAIL 2
2018-01-24 04:22:27 +03:00
typedef struct {
c3_y sat_y;
u3_noun a;
u3_noun b;
2018-01-24 04:22:27 +03:00
} eqframe;
static inline eqframe*
_eq_push(c3_ys mov, c3_ys off, u3_noun a, u3_noun b)
2018-01-24 04:22:27 +03:00
{
u3R->cap_p += mov;
eqframe* cur = u3to(eqframe, u3R->cap_p + off);
cur->sat_y = SONG_NONE;
cur->a = a;
cur->b = b;
return cur;
2018-01-24 04:22:27 +03:00
}
static inline eqframe*
_eq_pop(c3_ys mov, c3_ys off)
2018-01-24 04:22:27 +03:00
{
u3R->cap_p -= mov;
return u3to(eqframe, u3R->cap_p + off);
2018-01-24 04:22:27 +03:00
}
/* _sing_one(): do not pick a unified pointer for identical (a) and (b).
*/
static void
_sing_one(u3_noun* a, u3_noun* b)
{
// this space left intentionally blank
}
2014-10-09 02:16:24 +04:00
/* _sung_one(): pick a unified pointer for identical (a) and (b).
2017-05-11 13:57:18 +03:00
**
** Assumes exclusive access to noun memory.
*/
2014-10-09 02:16:24 +04:00
static void
_sung_one(u3_noun* a, u3_noun* b)
{
2017-05-11 13:57:18 +03:00
2014-10-09 02:16:24 +04:00
if ( *a == *b ) {
return;
2017-05-11 13:57:18 +03:00
} else {
u3_road* rod_u = u3R;
while ( 1 ) {
2014-10-09 02:16:24 +04:00
//
2017-05-11 13:57:18 +03:00
// we can't perform this kind of butchery on the home road,
// where asynchronous things can allocate.
//
if ( u3R == &u3H->rod_u ) {
break;
2014-10-09 02:16:24 +04:00
}
2017-05-11 13:57:18 +03:00
else {
c3_o asr_o = u3a_is_senior(u3R, *a);
c3_o bsr_o = u3a_is_senior(u3R, *b);
if ( _(asr_o) && _(bsr_o) ) {
//
// when unifying on a higher road, we can't free nouns,
// because we can't track junior nouns that point into
// that road.
//
// this is just an implementation issue -- we could set use
// counts to 0 without actually freeing. but the allocator
// would have to be actually designed for this.
//
// not freeing may generate spurious leaks, so we disable
// senior unification when debugging memory. this will
// cause a very slow boot process as the compiler compiles
// itself, constantly running into duplicates.
//
#ifdef U3_MEMORY_DEBUG
return;
#else
u3R = u3to(u3_road, u3R->par_p);
continue;
#endif
}
if ( _(asr_o) && !_(bsr_o) ){
if ( u3R == rod_u ) { u3z(*b); }
*b = *a;
}
if ( _(bsr_o) && !_(asr_o) ) {
if ( u3R == rod_u ) { u3z(*a); }
*a = *b;
}
if ( u3a_is_north(u3R) ) {
if ( *a <= *b ) {
u3k(*a);
if ( u3R == rod_u ) { u3z(*b); }
*b = *a;
} else {
u3k(*b);
if ( u3R == rod_u ) { u3z(*a); }
*a = *b;
}
}
else {
if ( *a >= *b ) {
u3k(*a);
if ( u3R == rod_u ) { u3z(*b); }
*b = *a;
} else {
u3k(*b);
if ( u3R == rod_u ) { u3z(*a); }
*a = *b;
}
}
break;
2014-10-09 02:16:24 +04:00
}
}
2017-05-11 13:57:18 +03:00
u3R = rod_u;
2014-10-09 02:16:24 +04:00
}
}
static inline c3_o
2018-01-29 21:52:47 +03:00
_song_atom(u3_atom a, u3_atom b)
{
u3a_atom* a_u = u3a_to_ptr(a);
if ( !_(u3a_is_atom(b)) ||
_(u3a_is_cat(a)) ||
_(u3a_is_cat(b)) )
{
return c3n;
}
else {
u3a_atom* b_u = u3a_to_ptr(b);
if ( a_u->mug_w &&
b_u->mug_w &&
(a_u->mug_w != b_u->mug_w) )
{
return c3n;
}
else {
c3_w w_rez = a_u->len_w;
c3_w w_mox = b_u->len_w;
if ( w_rez != w_mox ) {
return c3n;
}
else {
c3_w i_w;
for ( i_w = 0; i_w < w_rez; i_w++ ) {
if ( a_u->buf_w[i_w] != b_u->buf_w[i_w] ) {
return c3n;
}
}
}
}
}
return c3y;
}
/* _song_x_cape(): unifying equality with comparison deduplication
* (tightly coupled to _song_x)
*/
static c3_o
_song_x_cape(c3_ys mov, c3_ys off,
eqframe* fam, eqframe* don,
u3p(u3h_root) har_p,
void (*uni)(u3_noun*, u3_noun*))
{
u3_noun a, b, key;
u3_weak got;
u3a_cell* a_u;
u3a_cell* b_u;
while ( don != fam ) {
a = fam->a;
b = fam->b;
switch ( fam->sat_y ) {
case SONG_NONE:
if ( a == b ) {
break;
}
else if ( c3y == u3a_is_atom(a) ) {
if ( c3n == _song_atom(a, b) ) {
return c3n;
}
else {
break;
}
}
else if ( c3y == u3a_is_atom(b) ) {
return c3n;
}
else {
u3a_cell* a_u = u3a_to_ptr(a);
u3a_cell* b_u = u3a_to_ptr(b);
if ( a_u->mug_w &&
b_u->mug_w &&
(a_u->mug_w != b_u->mug_w) ) {
return c3n;
}
else {
key = u3nc(u3a_to_off(a), u3a_to_off(b));
u3t_off(euq_o);
got = u3h_get(har_p, key);
u3t_on(euq_o);
u3z(key);
if ( u3_none != got ) {
fam = _eq_pop(mov, off);
continue;
}
fam->sat_y = SONG_HEAD;
fam = _eq_push(mov, off, a_u->hed, b_u->hed);
continue;
}
}
case SONG_HEAD:
a_u = u3a_to_ptr(a);
b_u = u3a_to_ptr(b);
uni(&(a_u->hed), &(b_u->hed));
fam->sat_y = SONG_TAIL;
fam = _eq_push(mov, off, a_u->tel, b_u->tel);
continue;
case SONG_TAIL:
a_u = u3a_to_ptr(a);
b_u = u3a_to_ptr(b);
uni(&(a_u->tel), &(b_u->tel));
break;
default:
c3_assert(0);
break;
}
key = u3nc(u3a_to_off(a), u3a_to_off(b));
u3t_off(euq_o);
u3h_put(har_p, key, c3y);
u3t_on(euq_o);
u3z(key);
fam = _eq_pop(mov, off);
}
return c3y;
}
/* _song_x(): yes if a and b are the same noun, use uni to unify
2014-09-05 23:55:16 +04:00
*/
2014-11-06 06:10:22 +03:00
static c3_o
_song_x(u3_noun a, u3_noun b, void (*uni)(u3_noun*, u3_noun*))
2014-09-05 23:55:16 +04:00
{
u3p(eqframe) empty = u3R->cap_p;
c3_y wis_y = c3_wiseof(eqframe);
c3_o nor_o = u3a_is_north(u3R);
c3_ys mov = ( c3y == nor_o ? -wis_y : wis_y );
c3_ys off = ( c3y == nor_o ? 0 : -wis_y );
2018-03-16 02:58:04 +03:00
c3_s ovr_s = 0;
eqframe* fam = _eq_push(mov, off, a, b);
eqframe* don = u3to(eqframe, empty + off);
u3a_cell* a_u;
u3a_cell* b_u;
while ( don != fam ) {
a = fam->a;
b = fam->b;
switch ( fam->sat_y ) {
case SONG_NONE:
if ( a == b ) {
break;
}
else if ( c3y == u3a_is_atom(a) ) {
if ( c3n == _song_atom(a, b) ) {
u3R->cap_p = empty;
return c3n;
}
else {
break;
}
}
else if ( c3y == u3a_is_atom(b) ) {
u3R->cap_p = empty;
return c3n;
}
else {
a_u = u3a_to_ptr(a);
b_u = u3a_to_ptr(b);
2018-01-24 04:22:27 +03:00
if ( a_u->mug_w &&
b_u->mug_w &&
(a_u->mug_w != b_u->mug_w) ) {
u3R->cap_p = empty;
return c3n;
2014-09-05 23:55:16 +04:00
}
2018-01-24 04:22:27 +03:00
else {
fam->sat_y = SONG_HEAD;
fam = _eq_push(mov, off, a_u->hed, b_u->hed);
continue;
2014-09-05 23:55:16 +04:00
}
}
case SONG_HEAD:
a_u = u3a_to_ptr(a);
b_u = u3a_to_ptr(b);
uni(&(a_u->hed), &(b_u->hed));
fam->sat_y = SONG_TAIL;
fam = _eq_push(mov, off, a_u->tel, b_u->tel);
continue;
case SONG_TAIL:
a_u = u3a_to_ptr(a);
b_u = u3a_to_ptr(b);
uni(&(a_u->tel), &(b_u->tel));
break;
default:
c3_assert(0);
break;
2014-09-05 23:55:16 +04:00
}
2018-03-16 02:58:04 +03:00
if ( 0 == ++ovr_s ) {
u3p(u3h_root) har_p = u3h_new();
c3_o ret_o = _song_x_cape(mov, off, fam, don, har_p, uni);
u3h_free(har_p);
u3R->cap_p = empty;
return ret_o;
}
fam = _eq_pop(mov, off);
}
return c3y;
2014-09-05 23:55:16 +04:00
}
2015-06-08 22:48:35 +03:00
/* u3r_sang(): yes iff (a) and (b) are the same noun, unifying equals.
*/
c3_o
u3r_sang(u3_noun a, u3_noun b)
{
2018-03-06 00:39:11 +03:00
c3_o ret_o;
u3t_on(euq_o);
ret_o = _song_x(a, b, &_sang_one);
u3t_off(euq_o);
return ret_o;
2015-06-08 22:48:35 +03:00
}
2014-11-06 03:20:01 +03:00
/* u3r_sing():
2014-09-05 23:55:16 +04:00
**
** Yes iff (a) and (b) are the same noun.
*/
2014-11-06 06:10:22 +03:00
c3_o
2015-06-08 22:48:35 +03:00
u3r_sing(u3_noun a, u3_noun b)
2014-09-05 23:55:16 +04:00
{
2015-06-08 22:48:35 +03:00
#ifndef U3_MEMORY_DEBUG
if ( u3R->par_p ) {
2018-03-06 00:39:11 +03:00
return u3r_sang(a, b);
2015-06-08 22:48:35 +03:00
}
#endif
{
c3_o ret_o;
2015-01-15 22:10:29 +03:00
2015-06-08 22:48:35 +03:00
u3t_on(euq_o);
ret_o = _song_x(a, b, &_sing_one);
2015-06-08 22:48:35 +03:00
u3t_off(euq_o);
2015-01-15 22:10:29 +03:00
2015-06-08 22:48:35 +03:00
return ret_o;
}
2014-09-05 23:55:16 +04:00
}
2014-11-06 03:20:01 +03:00
/* u3r_sung(): yes iff (a) and (b) are the same noun, unifying equals.
2014-10-09 02:16:24 +04:00
*/
2014-11-06 06:10:22 +03:00
c3_o
2015-06-08 22:48:35 +03:00
u3r_sung(u3_noun a, u3_noun b)
2014-10-09 02:16:24 +04:00
{
2018-03-06 00:39:11 +03:00
c3_o ret_o;
u3t_on(euq_o);
ret_o = _song_x(a, b, &_sung_one);
u3t_off(euq_o);
return ret_o;
2014-10-09 02:16:24 +04:00
}
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_fing(u3_noun a,
2014-09-06 00:13:24 +04:00
u3_noun b)
2014-09-05 23:55:16 +04:00
{
2014-11-05 04:18:47 +03:00
return (a == b) ? c3y : c3n;
2014-09-05 23:55:16 +04:00
}
2014-11-06 03:20:01 +03:00
/* u3r_sing_cell():
2014-09-05 23:55:16 +04:00
**
** Yes iff `[p q]` and `b` are the same noun.
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_sing_cell(u3_noun p,
2014-09-06 00:13:24 +04:00
u3_noun q,
u3_noun b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 03:20:01 +03:00
return c3a(_(u3a_is_cell(b)),
c3a(u3r_sing(p, u3a_h(b)),
u3r_sing(q, u3a_t(b))));
2014-09-05 23:55:16 +04:00
}
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_fing_cell(u3_noun p,
2014-09-06 00:13:24 +04:00
u3_noun q,
u3_noun b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 03:20:01 +03:00
return c3a(_(u3a_is_cell(b)),
c3a(u3r_fing(p, u3a_h(b)),
u3r_fing(q, u3a_t(b))));
2014-09-05 23:55:16 +04:00
}
2014-11-06 03:20:01 +03:00
/* u3r_sing_mixt():
2014-09-05 23:55:16 +04:00
**
** Yes iff `[p q]` and `b` are the same noun.
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_sing_mixt(const c3_c* p_c,
2014-09-06 00:13:24 +04:00
u3_noun q,
u3_noun b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 03:20:01 +03:00
return c3a(_(u3a_is_cell(b)),
c3a(u3r_sing_c(p_c, u3a_h(b)),
u3r_sing(q, u3a_t(b))));
2014-09-05 23:55:16 +04:00
}
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_fing_mixt(const c3_c* p_c,
2014-09-06 00:13:24 +04:00
u3_noun q,
u3_noun b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 03:20:01 +03:00
return c3a(_(u3a_is_cell(b)),
c3a(u3r_sing_c(p_c, u3a_h(b)),
u3r_fing(q, u3a_t(b))));
2014-09-05 23:55:16 +04:00
}
2014-11-06 03:20:01 +03:00
/* u3r_sing_trel():
2014-09-05 23:55:16 +04:00
**
** Yes iff `[p q r]` and `b` are the same noun.
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_sing_trel(u3_noun p,
2014-09-06 00:13:24 +04:00
u3_noun q,
u3_noun r,
u3_noun b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 03:20:01 +03:00
return c3a(_(u3a_is_cell(b)),
c3a(u3r_sing(p, u3a_h(b)),
u3r_sing_cell(q, r, u3a_t(b))));
2014-09-05 23:55:16 +04:00
}
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_fing_trel(u3_noun p,
2014-09-06 00:13:24 +04:00
u3_noun q,
u3_noun r,
u3_noun b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 03:20:01 +03:00
return c3a(_(u3a_is_cell(b)),
c3a(u3r_fing(p, u3a_h(b)),
u3r_fing_cell(q, r, u3a_t(b))));
2014-09-05 23:55:16 +04:00
}
2014-11-06 03:20:01 +03:00
/* u3r_sing_qual():
2014-09-05 23:55:16 +04:00
**
** Yes iff `[p q r]` and `b` are the same noun.
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_sing_qual(u3_noun p,
2014-09-06 00:13:24 +04:00
u3_noun q,
u3_noun r,
u3_noun s,
u3_noun b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 03:20:01 +03:00
return c3a(_(u3a_is_cell(b)),
c3a(u3r_sing(p, u3a_h(b)),
u3r_sing_trel(q, r, s, u3a_t(b))));
2014-09-05 23:55:16 +04:00
}
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_fing_qual(u3_noun p,
2014-09-06 00:13:24 +04:00
u3_noun q,
u3_noun r,
u3_noun s,
u3_noun b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 03:20:01 +03:00
return c3a(_(u3a_is_cell(b)),
c3a(u3r_fing(p, u3a_h(b)),
u3r_fing_trel(q, r, s, u3a_t(b))));
2014-09-05 23:55:16 +04:00
}
2014-11-06 03:20:01 +03:00
/* u3r_nord():
2014-09-05 23:55:16 +04:00
**
** Return 0, 1 or 2 if `a` is below, equal to, or above `b`.
*/
2014-09-06 00:13:24 +04:00
u3_atom
2014-11-06 03:20:01 +03:00
u3r_nord(u3_noun a,
2014-09-06 00:13:24 +04:00
u3_noun b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 02:36:30 +03:00
c3_assert(u3_none != a);
c3_assert(u3_none != b);
2014-09-05 23:55:16 +04:00
if ( a == b ) {
return 1;
}
else {
2014-11-06 03:20:01 +03:00
if ( _(u3a_is_atom(a)) ) {
if ( !_(u3a_is_atom(b)) ) {
2014-09-05 23:55:16 +04:00
return 0;
} else {
2014-11-06 03:20:01 +03:00
if ( _(u3a_is_cat(a)) ) {
if ( _(u3a_is_cat(b)) ) {
2014-09-05 23:55:16 +04:00
return (a < b) ? 0 : 2;
}
else return 0;
}
2014-11-06 03:20:01 +03:00
else if ( _(u3a_is_cat(b)) ) {
2014-09-05 23:55:16 +04:00
return 2;
}
else {
2014-11-06 03:20:01 +03:00
u3a_atom* a_u = u3a_to_ptr(a);
u3a_atom* b_u = u3a_to_ptr(b);
2014-09-05 23:55:16 +04:00
c3_w w_rez = a_u->len_w;
c3_w w_mox = b_u->len_w;
if ( w_rez != w_mox ) {
return (w_rez < w_mox) ? 0 : 2;
}
else {
c3_w i_w;
for ( i_w = 0; i_w < w_rez; i_w++ ) {
c3_w ai_w = a_u->buf_w[i_w];
c3_w bi_w = b_u->buf_w[i_w];
if ( ai_w != bi_w ) {
return (ai_w < bi_w) ? 0 : 2;
}
}
return 1;
}
}
}
} else {
2014-11-06 03:20:01 +03:00
if ( _(u3a_is_atom(b)) ) {
2014-09-05 23:55:16 +04:00
return 2;
} else {
2014-11-06 03:20:01 +03:00
u3_atom c = u3r_nord(u3a_h(a), u3a_h(b));
2014-09-05 23:55:16 +04:00
if ( 1 == c ) {
2014-11-06 03:20:01 +03:00
return u3r_nord(u3a_t(a), u3a_t(b));
2014-09-05 23:55:16 +04:00
} else {
return c;
}
}
}
}
}
2014-11-06 03:20:01 +03:00
/* u3r_sing_c():
2014-09-05 23:55:16 +04:00
**
** Yes iff (b) is the same noun as the C string a_c.
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_sing_c(const c3_c* a_c,
2014-10-24 04:35:26 +04:00
u3_noun b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 02:36:30 +03:00
c3_assert(u3_none != b);
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( !_(u3a_is_atom(b)) ) {
2014-11-05 04:18:47 +03:00
return c3n;
2014-09-05 23:55:16 +04:00
}
else {
c3_w w_sof = strlen(a_c);
c3_w i_w;
2014-11-06 03:20:01 +03:00
if ( w_sof != u3r_met(3, b) ) {
2014-11-05 04:18:47 +03:00
return c3n;
2014-10-24 04:35:26 +04:00
}
2014-09-05 23:55:16 +04:00
for ( i_w = 0; i_w < w_sof; i_w++ ) {
2014-11-06 03:20:01 +03:00
if ( u3r_byte(i_w, b) != a_c[i_w] ) {
2014-11-05 04:18:47 +03:00
return c3n;
2014-09-05 23:55:16 +04:00
}
}
2014-11-05 04:18:47 +03:00
return c3y;
2014-09-05 23:55:16 +04:00
}
}
2014-11-06 03:20:01 +03:00
/* u3r_bush():
2014-09-05 23:55:16 +04:00
**
** Factor [a] as a bush [b.[p q] c].
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_bush(u3_noun a,
2014-09-06 00:13:24 +04:00
u3_noun* b,
u3_noun* c)
2014-09-05 23:55:16 +04:00
{
2014-11-06 02:36:30 +03:00
c3_assert(u3_none != a);
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( _(u3a_is_atom(a)) ) {
2014-11-05 04:18:47 +03:00
return c3n;
2014-09-05 23:55:16 +04:00
}
else {
2014-11-06 03:20:01 +03:00
*b = u3a_h(a);
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( _(u3a_is_atom(*b)) ) {
2014-11-05 04:18:47 +03:00
return c3n;
2014-09-05 23:55:16 +04:00
} else {
2014-11-06 03:20:01 +03:00
*c = u3a_t(a);
2014-11-05 04:18:47 +03:00
return c3y;
2014-09-05 23:55:16 +04:00
}
}
}
2014-11-06 03:20:01 +03:00
/* u3r_cell():
2014-09-05 23:55:16 +04:00
**
** Factor (a) as a cell (b c).
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_cell(u3_noun a,
2014-09-06 00:13:24 +04:00
u3_noun* b,
u3_noun* c)
2014-09-05 23:55:16 +04:00
{
2014-11-06 02:36:30 +03:00
c3_assert(u3_none != a);
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( _(u3a_is_atom(a)) ) {
2014-11-05 04:18:47 +03:00
return c3n;
2014-09-05 23:55:16 +04:00
}
else {
2014-11-06 03:20:01 +03:00
if ( b ) *b = u3a_h(a);
if ( c ) *c = u3a_t(a);
2014-11-05 04:18:47 +03:00
return c3y;
2014-09-05 23:55:16 +04:00
}
}
2014-11-06 03:20:01 +03:00
/* u3r_p():
2014-09-05 23:55:16 +04:00
**
** & [0] if [a] is of the form [b *c].
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_p(u3_noun a,
2014-09-06 00:13:24 +04:00
u3_noun b,
u3_noun* c)
2014-09-05 23:55:16 +04:00
{
2014-09-06 00:13:24 +04:00
u3_noun feg, nux;
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( (c3y == u3r_cell(a, &feg, &nux)) &&
(c3y == u3r_sing(feg, b)) )
2014-09-05 23:55:16 +04:00
{
*c = nux;
2014-11-05 04:18:47 +03:00
return c3y;
2014-09-05 23:55:16 +04:00
}
2014-11-05 04:18:47 +03:00
else return c3n;
2014-09-05 23:55:16 +04:00
}
2014-11-06 03:20:01 +03:00
/* u3r_pq():
2014-09-05 23:55:16 +04:00
**
** & [0] if [a] is of the form [b *c d].
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_pq(u3_noun a,
2014-09-06 00:13:24 +04:00
u3_noun b,
u3_noun* c,
u3_noun* d)
2014-09-05 23:55:16 +04:00
{
2014-09-06 00:13:24 +04:00
u3_noun nux;
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( (c3y == u3r_p(a, b, &nux)) &&
(c3y == u3r_cell(nux, c, d)) )
2014-09-05 23:55:16 +04:00
{
2014-11-05 04:18:47 +03:00
return c3y;
2014-09-05 23:55:16 +04:00
}
2014-11-05 04:18:47 +03:00
else return c3n;
2014-09-05 23:55:16 +04:00
}
2014-11-06 03:20:01 +03:00
/* u3r_pqr():
2014-09-05 23:55:16 +04:00
**
** & [0] if [a] is of the form [b *c *d *e].
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_pqr(u3_noun a,
2014-09-06 00:13:24 +04:00
u3_noun b,
u3_noun* c,
u3_noun* d,
u3_noun* e)
2014-09-05 23:55:16 +04:00
{
2014-09-06 00:13:24 +04:00
u3_noun nux;
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( (c3y == u3r_p(a, b, &nux)) &&
(c3y == u3r_trel(nux, c, d, e)) )
2014-09-05 23:55:16 +04:00
{
2014-11-05 04:18:47 +03:00
return c3y;
2014-09-05 23:55:16 +04:00
}
2014-11-05 04:18:47 +03:00
else return c3n;
2014-09-05 23:55:16 +04:00
}
2014-11-06 03:20:01 +03:00
/* u3r_pqrs():
2014-09-05 23:55:16 +04:00
**
** & [0] if [a] is of the form [b *c *d *e *f].
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_pqrs(u3_noun a,
2014-09-06 00:13:24 +04:00
u3_noun b,
u3_noun* c,
u3_noun* d,
u3_noun* e,
u3_noun* f)
2014-09-05 23:55:16 +04:00
{
2014-09-06 00:13:24 +04:00
u3_noun nux;
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( (c3y == u3r_p(a, b, &nux)) &&
(c3y == u3r_qual(nux, c, d, e, f)) )
2014-09-05 23:55:16 +04:00
{
2014-11-05 04:18:47 +03:00
return c3y;
2014-09-05 23:55:16 +04:00
}
2014-11-05 04:18:47 +03:00
else return c3n;
2014-09-05 23:55:16 +04:00
}
2014-11-06 03:20:01 +03:00
/* u3r_trel():
2014-09-05 23:55:16 +04:00
**
** Factor (a) as a trel (b c d).
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_trel(u3_noun a,
2014-09-06 00:13:24 +04:00
u3_noun *b,
u3_noun *c,
u3_noun *d)
2014-09-05 23:55:16 +04:00
{
2014-09-06 00:13:24 +04:00
u3_noun guf;
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( (c3y == u3r_cell(a, b, &guf)) &&
(c3y == u3r_cell(guf, c, d)) ) {
2014-11-05 04:18:47 +03:00
return c3y;
2014-09-05 23:55:16 +04:00
}
else {
2014-11-05 04:18:47 +03:00
return c3n;
2014-09-05 23:55:16 +04:00
}
}
2014-11-06 03:20:01 +03:00
/* u3r_qual():
2014-09-05 23:55:16 +04:00
**
** Factor (a) as a qual (b c d e).
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_qual(u3_noun a,
2014-09-06 00:13:24 +04:00
u3_noun* b,
u3_noun* c,
u3_noun* d,
u3_noun* e)
2014-09-05 23:55:16 +04:00
{
2014-09-06 00:13:24 +04:00
u3_noun guf;
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( (c3y == u3r_cell(a, b, &guf)) &&
(c3y == u3r_trel(guf, c, d, e)) ) {
2014-11-05 04:18:47 +03:00
return c3y;
2014-09-05 23:55:16 +04:00
}
2014-11-05 04:18:47 +03:00
else return c3n;
2014-09-05 23:55:16 +04:00
}
2014-11-06 03:20:01 +03:00
/* u3r_quil():
2014-09-05 23:55:16 +04:00
**
** Factor (a) as a quil (b c d e f).
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_quil(u3_noun a,
2014-09-06 00:13:24 +04:00
u3_noun* b,
u3_noun* c,
u3_noun* d,
u3_noun* e,
u3_noun* f)
2014-09-05 23:55:16 +04:00
{
2014-09-06 00:13:24 +04:00
u3_noun guf;
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( (c3y == u3r_cell(a, b, &guf)) &&
(c3y == u3r_qual(guf, c, d, e, f)) ) {
2014-11-05 04:18:47 +03:00
return c3y;
2014-09-05 23:55:16 +04:00
}
2014-11-05 04:18:47 +03:00
else return c3n;
2014-09-05 23:55:16 +04:00
}
2014-11-06 03:20:01 +03:00
/* u3r_hext():
2014-09-05 23:55:16 +04:00
**
** Factor (a) as a hext (b c d e f g)
*/
2014-11-06 06:10:22 +03:00
c3_o
2014-11-06 03:20:01 +03:00
u3r_hext(u3_noun a,
2014-09-06 00:13:24 +04:00
u3_noun* b,
u3_noun* c,
u3_noun* d,
u3_noun* e,
u3_noun* f,
u3_noun* g)
2014-09-05 23:55:16 +04:00
{
2014-09-06 00:13:24 +04:00
u3_noun guf;
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( (c3y == u3r_cell(a, b, &guf)) &&
(c3y == u3r_quil(guf, c, d, e, f, g)) ) {
2014-11-05 04:18:47 +03:00
return c3y;
2014-09-05 23:55:16 +04:00
}
2014-11-05 04:18:47 +03:00
else return c3n;
2014-09-05 23:55:16 +04:00
}
2014-11-06 03:20:01 +03:00
/* u3r_met():
2014-09-05 23:55:16 +04:00
**
** Return the size of (b) in bits, rounded up to
** (1 << a_y).
**
** For example, (a_y == 3) returns the size in bytes.
*/
c3_w
2014-11-06 03:20:01 +03:00
u3r_met(c3_y a_y,
2014-09-06 00:13:24 +04:00
u3_atom b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 02:36:30 +03:00
c3_assert(u3_none != b);
2014-11-06 03:20:01 +03:00
c3_assert(_(u3a_is_atom(b)));
2014-09-05 23:55:16 +04:00
if ( b == 0 ) {
return 0;
}
else {
/* gal_w: number of words besides (daz_w) in (b).
** daz_w: top word in (b).
*/
c3_w gal_w;
c3_w daz_w;
2014-11-06 03:20:01 +03:00
if ( _(u3a_is_cat(b)) ) {
2014-09-05 23:55:16 +04:00
gal_w = 0;
daz_w = b;
}
else {
2014-11-06 03:20:01 +03:00
u3a_atom* b_u = u3a_to_ptr(b);
2014-09-05 23:55:16 +04:00
gal_w = (b_u->len_w) - 1;
daz_w = b_u->buf_w[gal_w];
}
switch ( a_y ) {
case 0:
case 1:
case 2: {
/* col_w: number of bits in (daz_w)
** bif_w: number of bits in (b)
*/
c3_w bif_w, col_w;
col_w = c3_bits_word(daz_w);
bif_w = col_w + (gal_w << 5);
return (bif_w + ((1 << a_y) - 1)) >> a_y;
}
case 3: {
return (gal_w << 2)
+ ((daz_w >> 24) ? 4 : (daz_w >> 16) ? 3 : (daz_w >> 8) ? 2 : 1);
}
case 4: {
return (gal_w << 1)
+ ((daz_w >> 16) ? 2 : 1);
}
default: {
c3_y gow_y = (a_y - 5);
return ((gal_w + 1) + ((1 << gow_y) - 1)) >> gow_y;
}
}
}
}
2014-11-06 03:20:01 +03:00
/* u3r_bit():
2014-09-05 23:55:16 +04:00
**
** Return bit (a_w) of (b).
*/
c3_b
2014-11-06 03:20:01 +03:00
u3r_bit(c3_w a_w,
2014-09-06 00:13:24 +04:00
u3_atom b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 02:36:30 +03:00
c3_assert(u3_none != b);
2014-11-06 03:20:01 +03:00
c3_assert(_(u3a_is_atom(b)));
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( _(u3a_is_cat(b)) ) {
2014-09-05 23:55:16 +04:00
if ( a_w >= 31 ) {
return 0;
}
else return (1 & (b >> a_w));
}
else {
2014-11-06 03:20:01 +03:00
u3a_atom* b_u = u3a_to_ptr(b);
2014-09-05 23:55:16 +04:00
c3_y vut_y = (a_w & 31);
c3_w pix_w = (a_w >> 5);
if ( pix_w >= b_u->len_w ) {
return 0;
}
else {
c3_w nys_w = b_u->buf_w[pix_w];
return (1 & (nys_w >> vut_y));
}
}
}
2014-11-06 03:20:01 +03:00
/* u3r_byte():
2014-09-05 23:55:16 +04:00
**
** Return byte (a_w) of (b).
*/
c3_y
2014-11-06 03:20:01 +03:00
u3r_byte(c3_w a_w,
2014-09-06 00:13:24 +04:00
u3_atom b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 02:36:30 +03:00
c3_assert(u3_none != b);
2014-11-06 03:20:01 +03:00
c3_assert(_(u3a_is_atom(b)));
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( _(u3a_is_cat(b)) ) {
2014-09-05 23:55:16 +04:00
if ( a_w > 3 ) {
return 0;
}
else return (255 & (b >> (a_w << 3)));
}
else {
2014-11-06 03:20:01 +03:00
u3a_atom* b_u = u3a_to_ptr(b);
2015-07-29 23:43:55 +03:00
c3_y vut_y = (a_w & 3);
c3_w pix_w = (a_w >> 2);
2014-09-05 23:55:16 +04:00
if ( pix_w >= b_u->len_w ) {
return 0;
}
else {
c3_w nys_w = b_u->buf_w[pix_w];
return (255 & (nys_w >> (vut_y << 3)));
}
}
}
2014-11-06 03:20:01 +03:00
/* u3r_bytes():
2014-09-05 23:55:16 +04:00
**
** Copy bytes (a_w) through (a_w + b_w - 1) from (d) to (c).
*/
void
2014-11-06 03:20:01 +03:00
u3r_bytes(c3_w a_w,
2014-09-05 23:55:16 +04:00
c3_w b_w,
c3_y* c_y,
2014-09-06 00:13:24 +04:00
u3_atom d)
2014-09-05 23:55:16 +04:00
{
2014-11-06 02:36:30 +03:00
c3_assert(u3_none != d);
2015-07-29 23:43:55 +03:00
c3_assert(_(u3a_is_atom(d)));
if ( _(u3a_is_cat(d)) ) {
c3_w e_w = d >> (c3_min(a_w, 4) << 3);
c3_w m_w = c3_min(b_w, 4);
memcpy(c_y, (c3_y*)&e_w, m_w);
if ( b_w > 4 ) {
memset(c_y + 4, 0, b_w - 4);
}
}
else {
u3a_atom* d_u = u3a_to_ptr(d);
c3_w n_w = d_u->len_w << 2;
c3_y* x_y = (c3_y*)d_u->buf_w + a_w;
2014-09-05 23:55:16 +04:00
2015-07-29 23:43:55 +03:00
if ( a_w >= n_w ) {
memset(c_y, 0, b_w);
}
else {
c3_w z_w = c3_min(b_w, n_w - a_w);
memcpy(c_y, x_y, z_w);
if ( b_w > n_w - a_w ) {
memset(c_y + z_w, 0, b_w + a_w - n_w);
}
}
2014-09-05 23:55:16 +04:00
}
}
2014-11-06 03:20:01 +03:00
/* u3r_mp():
2014-09-05 23:55:16 +04:00
**
** Copy (b) into (a_mp).
*/
void
2014-11-06 03:20:01 +03:00
u3r_mp(mpz_t a_mp,
2014-09-06 00:13:24 +04:00
u3_atom b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 02:36:30 +03:00
c3_assert(u3_none != b);
2014-11-06 03:20:01 +03:00
c3_assert(_(u3a_is_atom(b)));
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( _(u3a_is_cat(b)) ) {
2014-09-05 23:55:16 +04:00
mpz_init_set_ui(a_mp, b);
}
else {
2014-11-06 03:20:01 +03:00
u3a_atom* b_u = u3a_to_ptr(b);
2014-09-05 23:55:16 +04:00
c3_w len_w = b_u->len_w;
/* Slight deficiency in the GMP API.
*/
c3_assert(!(len_w >> 27));
mpz_init2(a_mp, len_w << 5);
/* Efficiency: horrible.
*/
{
c3_w *buf_w = alloca(len_w << 2);
c3_w i_w;
for ( i_w=0; i_w < len_w; i_w++ ) {
buf_w[i_w] = b_u->buf_w[i_w];
}
mpz_import(a_mp, len_w, -1, 4, 0, 0, buf_w);
}
}
}
2014-11-06 03:20:01 +03:00
/* u3r_word():
2014-09-05 23:55:16 +04:00
**
** Return word (a_w) of (b).
*/
c3_w
2014-11-06 03:20:01 +03:00
u3r_word(c3_w a_w,
2014-09-06 00:13:24 +04:00
u3_atom b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 02:36:30 +03:00
c3_assert(u3_none != b);
2014-11-06 03:20:01 +03:00
c3_assert(_(u3a_is_atom(b)));
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( _(u3a_is_cat(b)) ) {
2014-09-05 23:55:16 +04:00
if ( a_w > 0 ) {
return 0;
}
else return b;
}
else {
2014-11-06 03:20:01 +03:00
u3a_atom* b_u = u3a_to_ptr(b);
2014-09-05 23:55:16 +04:00
if ( a_w >= b_u->len_w ) {
return 0;
}
else return b_u->buf_w[a_w];
}
}
2014-11-06 03:20:01 +03:00
/* u3r_chub():
2014-09-05 23:55:16 +04:00
**
** Return double-word (a_w) of (b).
*/
c3_d
2014-11-06 03:20:01 +03:00
u3r_chub(c3_w a_w,
2014-09-06 00:13:24 +04:00
u3_atom b)
2014-09-05 23:55:16 +04:00
{
2014-11-06 03:20:01 +03:00
c3_w wlo_w = u3r_word(a_w * 2, b);
c3_w whi_w = u3r_word(1 + (a_w * 2), b);
2014-09-05 23:55:16 +04:00
return (((uint64_t)whi_w) << 32ULL) | ((uint64_t)wlo_w);
}
2014-11-06 03:20:01 +03:00
/* u3r_words():
2014-09-05 23:55:16 +04:00
**
** Copy words (a_w) through (a_w + b_w - 1) from (d) to (c).
*/
void
2014-11-06 03:20:01 +03:00
u3r_words(c3_w a_w,
2014-09-05 23:55:16 +04:00
c3_w b_w,
c3_w* c_w,
2014-09-06 00:13:24 +04:00
u3_atom d)
2014-09-05 23:55:16 +04:00
{
2014-11-06 02:36:30 +03:00
c3_assert(u3_none != d);
2015-07-29 23:43:55 +03:00
c3_assert(_(u3a_is_atom(d)));
2014-09-05 23:55:16 +04:00
2015-07-29 23:43:55 +03:00
if ( b_w == 0 ) {
return;
}
if ( _(u3a_is_cat(d)) ) {
if ( a_w == 0 ) {
*c_w = d;
memset((c3_y*)(c_w + 1), 0, (b_w - 1) << 2);
}
else {
memset((c3_y*)c_w, 0, b_w << 2);
}
}
else {
u3a_atom* d_u = u3a_to_ptr(d);
if ( a_w >= d_u->len_w ) {
memset((c3_y*)c_w, 0, b_w << 2);
}
else {
c3_w z_w = c3_min(b_w, d_u->len_w - a_w);
c3_w* x_w = d_u->buf_w + a_w;
memcpy((c3_y*)c_w, (c3_y*)x_w, z_w << 2);
if ( b_w > d_u->len_w - a_w ) {
memset((c3_y*)(c_w + z_w), 0, (b_w + a_w - d_u->len_w) << 2);
}
}
2014-09-05 23:55:16 +04:00
}
}
2019-01-10 21:15:01 +03:00
/* u3r_chubs():
**
** Copy double-words (a_w) through (a_w + b_w - 1) from (d) to (c).
*/
void
u3r_chubs(c3_w a_w,
c3_w b_w,
c3_d* c_d,
u3_atom d)
{
/* XX: assumes little-endian
*/
u3r_words(a_w * 2, b_w * 2, (c3_w *)c_d, d);
}
2014-11-06 03:20:01 +03:00
/* u3r_chop():
2014-09-05 23:55:16 +04:00
**
** Into the bloq space of `met`, from position `fum` for a
** span of `wid`, to position `tou`, XOR from atom `src`
** into `dst_w`.
*/
void
2014-11-06 03:20:01 +03:00
u3r_chop(c3_g met_g,
2014-09-05 23:55:16 +04:00
c3_w fum_w,
c3_w wid_w,
c3_w tou_w,
c3_w* dst_w,
2014-09-06 00:13:24 +04:00
u3_atom src)
2014-09-05 23:55:16 +04:00
{
c3_w i_w;
c3_w len_w;
c3_w* buf_w;
2014-11-06 02:36:30 +03:00
c3_assert(u3_none != src);
2014-11-06 03:20:01 +03:00
c3_assert(_(u3a_is_atom(src)));
2014-09-05 23:55:16 +04:00
2014-11-06 03:20:01 +03:00
if ( _(u3a_is_cat(src)) ) {
2014-09-05 23:55:16 +04:00
len_w = src ? 1 : 0;
buf_w = &src;
}
else {
2014-11-06 03:20:01 +03:00
u3a_atom* src_u = u3a_to_ptr(src);
2014-09-05 23:55:16 +04:00
len_w = src_u->len_w;
buf_w = src_u->buf_w;
}
if ( met_g < 5 ) {
c3_w san_w = (1 << met_g);
c3_w mek_w = ((1 << san_w) - 1);
c3_w baf_w = (fum_w << met_g);
c3_w bat_w = (tou_w << met_g);
// XX: efficiency: poor. Iterate by words.
//
for ( i_w = 0; i_w < wid_w; i_w++ ) {
c3_w waf_w = (baf_w >> 5);
c3_g raf_g = (baf_w & 31);
c3_w wat_w = (bat_w >> 5);
c3_g rat_g = (bat_w & 31);
c3_w hop_w;
hop_w = (waf_w >= len_w) ? 0 : buf_w[waf_w];
hop_w = (hop_w >> raf_g) & mek_w;
dst_w[wat_w] ^= (hop_w << rat_g);
baf_w += san_w;
bat_w += san_w;
}
}
else {
c3_g hut_g = (met_g - 5);
c3_w san_w = (1 << hut_g);
c3_w j_w;
for ( i_w = 0; i_w < wid_w; i_w++ ) {
c3_w wuf_w = (fum_w + i_w) << hut_g;
c3_w wut_w = (tou_w + i_w) << hut_g;
for ( j_w = 0; j_w < san_w; j_w++ ) {
dst_w[wut_w + j_w] ^=
((wuf_w + j_w) >= len_w)
? 0
: buf_w[wuf_w + j_w];
}
}
}
}
2014-11-06 03:20:01 +03:00
/* u3r_string(): `a` as malloced C string.
2014-09-05 23:55:16 +04:00
*/
c3_c*
2014-11-06 03:20:01 +03:00
u3r_string(u3_atom a)
2014-09-05 23:55:16 +04:00
{
2014-11-06 03:20:01 +03:00
c3_w met_w = u3r_met(3, a);
2014-09-05 23:55:16 +04:00
c3_c* str_c = c3_malloc(met_w + 1);
2014-11-06 03:20:01 +03:00
u3r_bytes(0, met_w, (c3_y*)str_c, a);
2014-09-05 23:55:16 +04:00
str_c[met_w] = 0;
return str_c;
}
2014-11-06 03:20:01 +03:00
/* u3r_tape(): `a`, a list of bytes, as malloced C string.
2014-09-05 23:55:16 +04:00
*/
c3_y*
2014-11-06 03:20:01 +03:00
u3r_tape(u3_noun a)
2014-09-05 23:55:16 +04:00
{
2014-09-06 00:13:24 +04:00
u3_noun b;
2014-09-05 23:55:16 +04:00
c3_w i_w;
c3_y *a_y;
2014-11-06 03:20:01 +03:00
for ( i_w = 0, b=a; c3y == u3a_is_cell(b); i_w++, b=u3a_t(b) )
2014-09-05 23:55:16 +04:00
;
a_y = c3_malloc(i_w + 1);
2014-11-06 03:20:01 +03:00
for ( i_w = 0, b=a; c3y == u3a_is_cell(b); i_w++, b=u3a_t(b) ) {
a_y[i_w] = u3a_h(b);
2014-09-05 23:55:16 +04:00
}
a_y[i_w] = 0;
return a_y;
}
2019-01-08 21:28:04 +03:00
/* u3r_mug_bytes(): Compute the mug of `buf`, `len`, LSW first.
*/
c3_w
u3r_mug_bytes(const c3_y *buf_y,
c3_w len_w)
{
c3_w syd_w = 0xcafebabe;
c3_w ham_w = 0;
while ( 0 == ham_w ) {
c3_w haz_w;
MurmurHash3_x86_32(buf_y, len_w, syd_w, &haz_w);
ham_w = (haz_w >> 31) ^ (haz_w & 0x7fffffff);
syd_w++;
}
return ham_w;
}
2019-01-08 21:28:04 +03:00
/* u3r_mug_chub(): Compute the mug of `num`, LSW first.
*/
c3_w
2019-01-08 21:28:04 +03:00
u3r_mug_chub(c3_d num_d)
{
c3_w buf_w[2];
buf_w[0] = (c3_w)(num_d & 0xffffffffULL);
buf_w[1] = (c3_w)(num_d >> 32ULL);
return u3r_mug_words(buf_w, 2);
}
2019-01-08 21:28:04 +03:00
/* u3r_mug_string(): Compute the mug of `a`, LSB first.
*/
c3_w
u3r_mug_string(const c3_c *a_c)
{
return u3r_mug_bytes((c3_y*)a_c, strlen(a_c));
}
/* u3r_mug_words(): 31-bit nonzero MurmurHash3 on raw words.
*/
c3_w
u3r_mug_words(const c3_w* key_w, c3_w len_w)
{
c3_w byt_w = 0;
c3_w wor_w;
while ( 0 < len_w ) {
wor_w = key_w[--len_w];
byt_w += _(u3a_is_cat(wor_w)) ? u3r_met(3, wor_w) : 4;
}
return u3r_mug_bytes((c3_y*)key_w, byt_w);
2019-01-08 21:28:04 +03:00
}
/* u3r_mug_both(): Join two mugs.
2014-09-07 02:39:28 +04:00
*/
c3_w
u3r_mug_both(c3_w lef_w, c3_w rit_w)
2014-09-07 02:39:28 +04:00
{
c3_w ham_w = lef_w ^ (0x7fffffff ^ rit_w);
return u3r_mug_words(&ham_w, (0 == ham_w) ? 0 : 1);
2014-09-07 02:39:28 +04:00
}
2019-01-08 21:28:04 +03:00
/* u3r_mug_cell(): Compute the mug of the cell `[hed tel]`.
*/
c3_w
u3r_mug_cell(u3_noun hed,
u3_noun tel)
{
c3_w lus_w = u3r_mug(hed);
c3_w biq_w = u3r_mug(tel);
return u3r_mug_both(lus_w, biq_w);
}
2019-01-16 05:08:21 +03:00
// mugframe: head and tail mugs of veb, 0 if uncalculated
//
typedef struct mugframe
{
u3_noun veb;
c3_w a;
c3_w b;
} mugframe;
static inline mugframe*
_mug_push(c3_ys mov, c3_ys off, u3_noun veb)
{
u3R->cap_p += mov;
// ensure we haven't overflowed the stack
// (off==0 means we're on a north road)
//
if ( 0 == off ) {
c3_assert(u3R->cap_p > u3R->hat_p);
}
else {
c3_assert(u3R->cap_p < u3R->hat_p);
}
2019-01-16 05:08:21 +03:00
mugframe* cur = u3to(mugframe, u3R->cap_p + off);
cur->veb = veb;
cur->a = 0;
cur->b = 0;
return cur;
}
static inline mugframe*
_mug_pop(c3_ys mov, c3_ys off, c3_w mug_w)
{
u3R->cap_p -= mov;
mugframe* fam = u3to(mugframe, u3R->cap_p + off);
// the bottom of the stack
//
if ( u3_none == fam->veb ) {
return fam;
}
2019-01-16 05:08:21 +03:00
// place return value in head of previous frame if not already calculated
//
if ( 0 == fam->a ) {
fam->a = mug_w;
}
// otherwise, place the return value in the tail
//
else if ( 0 == fam->b ) {
fam->b = mug_w;
}
// shouldn't reach
//
2019-01-16 05:08:21 +03:00
else {
2019-01-16 05:58:28 +03:00
c3_assert(0);
2019-01-16 05:08:21 +03:00
}
return fam;
}
// _mug_cat(): return the mug of a direct atom
//
static c3_w
_mug_cat(u3_atom veb)
{
c3_w len_w = u3r_met(3, veb);
return u3r_mug_bytes((c3_y*)&veb, len_w);
}
/* _mug_pug(): statefully mug an indirect atom
*/
static c3_w
_mug_pug(u3_atom veb)
{
u3a_atom* vat_u = (u3a_atom*)(u3a_to_ptr(veb));
c3_w len_w = u3r_met(3, veb);
c3_w mug_w = u3r_mug_bytes((c3_y*)vat_u->buf_w, len_w);
vat_u->mug_w = mug_w;
return mug_w;
}
2019-01-16 06:29:11 +03:00
/* _mug_atom(): mug an atom, either direct or indirect
*/
static c3_w
_mug_atom(u3_atom veb)
2019-01-16 05:08:21 +03:00
{
if ( _(u3a_is_cat(veb)) ) {
return _mug_cat(veb);
}
2019-01-16 06:29:11 +03:00
else {
2019-01-16 05:08:21 +03:00
return _mug_pug(veb);
}
2019-01-16 06:29:11 +03:00
}
2019-01-16 05:08:21 +03:00
2019-01-16 08:09:05 +03:00
// u3r_mug(): statefully mug a noun using a 31-bit MurmurHash3
2019-01-16 06:29:11 +03:00
//
c3_w
u3r_mug(u3_noun veb)
{
c3_assert( u3_none != veb );
if ( _(u3a_is_atom(veb)) ) {
return _mug_atom(veb);
2019-01-16 05:08:21 +03:00
}
c3_y wis_y = c3_wiseof(mugframe);
c3_o nor_o = u3a_is_north(u3R);
c3_ys mov = ( c3y == nor_o ? -wis_y : wis_y );
c3_ys off = ( c3y == nor_o ? 0 : -wis_y );
// stash the current stack pointer
//
2019-01-16 05:08:21 +03:00
u3p(mugframe) empty = u3R->cap_p;
// set the bottom of our stack
//
mugframe* don = _mug_push(mov, off, u3_none);
2019-01-16 05:58:28 +03:00
mugframe* fam = _mug_push(mov, off, veb);
2019-01-16 05:08:21 +03:00
c3_w mug_w;
c3_w a;
c3_w b;
2019-01-16 06:29:11 +03:00
u3a_noun* veb_u;
u3_noun hed, tal;
2019-01-16 05:08:21 +03:00
while ( don != fam ) {
a = fam->a;
b = fam->b;
veb = fam->veb;
veb_u = u3a_to_ptr(veb);
c3_assert(_(u3a_is_cell(veb)));
2019-01-16 06:29:11 +03:00
// already mugged; pop stack
2019-01-16 05:08:21 +03:00
//
if ( veb_u->mug_w ) {
mug_w = veb_u->mug_w;
2019-01-16 05:08:21 +03:00
fam = _mug_pop(mov, off, mug_w);
}
// neither head nor tail are mugged; start with head
2019-01-16 05:08:21 +03:00
//
else if ( 0 == a ) {
hed = u3h(veb);
if ( _(u3a_is_atom(hed)) ) {
fam->a = _mug_atom(hed);
2019-01-16 06:42:40 +03:00
}
else {
fam = _mug_push(mov, off, hed);
2019-01-16 06:42:40 +03:00
}
2019-01-16 05:08:21 +03:00
}
// head is mugged, but not tail; mug tail or push tail onto stack
2019-01-16 05:08:21 +03:00
//
else if ( 0 == b ) {
tal = u3t(veb);
if ( _(u3a_is_atom(tal)) ) {
fam->b = _mug_atom(tal);
2019-01-16 06:29:11 +03:00
}
else {
fam = _mug_push(mov, off, tal);
2019-01-16 06:29:11 +03:00
}
2019-01-16 05:08:21 +03:00
}
// both head and tail are mugged; combine them and pop stack
//
else {
mug_w = u3r_mug_both(a, b);
veb_u->mug_w = mug_w;
fam = _mug_pop(mov, off, mug_w);
}
2019-01-16 05:08:21 +03:00
}
2019-01-16 05:08:21 +03:00
u3R->cap_p = empty;
return mug_w;
}