X-Git-Url: https://pintos-os.org/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=src%2Fmath%2Fcategoricals.c;h=e1bbe1f99b141994365d32de6f888a2dee2619e4;hb=8180c5dd1591446174c0753ee960921786113403;hp=2bc2b626b0636c36c59a77fe0fb290738b87ead5;hpb=e0bcf133ab9091a2fa5227d2767651d8ec58caef;p=pspp diff --git a/src/math/categoricals.c b/src/math/categoricals.c index 2bc2b626b0..e1bbe1f99b 100644 --- a/src/math/categoricals.c +++ b/src/math/categoricals.c @@ -1,5 +1,5 @@ /* PSPP - a program for statistical analysis. - Copyright (C) 2009, 2010, 2011, 2012 Free Software Foundation, Inc. + Copyright (C) 2009, 2010, 2011, 2012, 2014 Free Software Foundation, Inc. This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by @@ -19,6 +19,7 @@ #include "math/categoricals.h" #include "math/interaction.h" +#include #include #include "data/case.h" @@ -34,8 +35,6 @@ #define CATEGORICALS_DEBUG 0 -#define EFFECTS_CODING 1 - struct value_node { struct hmap_node node; /* Node in hash map. */ @@ -45,13 +44,15 @@ struct value_node int index; /* A zero based unique index for this value */ }; + struct interaction_value { struct hmap_node node; /* Node in hash map */ - struct ccase *ccase; /* A case (probably the first in the dataset) which matches this value */ + struct ccase *ccase; /* A case (probably the first in the dataset) which matches + this value */ - double cc; /* Total of the weights of cases matching this interaction */ + double cc; /* Total of the weights of cases matching this interaction */ void *user_data; /* A pointer to data which the caller can store stuff */ }; @@ -65,7 +66,7 @@ lookup_value (const struct hmap *map, const union value *val, unsigned int hash, if (value_equal (&vn->val, val, width)) break; } - + return vn; } @@ -78,6 +79,22 @@ struct variable_node int n_vals; /* Number of values for this variable */ }; + +/* Comparison function to sort value_nodes in ascending order */ +static int +compare_value_node_3way (const void *vn1_, const void *vn2_, const void *aux) +{ + const struct value_node *const *vn1p = vn1_; + const struct value_node *const *vn2p = vn2_; + + const struct variable_node *vn = aux; + + + return value_compare_3way (&(*vn1p)->val, &(*vn2p)->val, var_get_width (vn->var)); +} + + + static struct variable_node * lookup_variable (const struct hmap *map, const struct variable *var, unsigned int hash) { @@ -86,17 +103,17 @@ lookup_variable (const struct hmap *map, const struct variable *var, unsigned in { if (vn->var == var) break; - - fprintf (stderr, "Warning: Hash table collision\n"); + + fprintf (stderr, "%s:%d Warning: Hash table collision\n", __FILE__, __LINE__); } - + return vn; } struct interact_params { - /* A map indexed by a interaction_value */ + /* A map of cases indexed by a interaction_value */ struct hmap ivmap; const struct interaction *iact; @@ -108,9 +125,9 @@ struct interact_params int n_cats; /* An array of integers df_n * df_{n-1} * df_{n-2} ... - These are the products of the degrees of freedom for the current - variable and all preceeding variables */ - int *df_prod; + These are the products of the degrees of freedom for the current + variable and all preceding variables */ + int *df_prod; double *enc_sum; @@ -179,6 +196,31 @@ struct categoricals const struct payload *payload; }; + +bool +categoricals_isbalanced (const struct categoricals *cat) +{ + int i; + + for (i = 0 ; i < cat->n_iap; ++i) + { + int v; + const struct interact_params *iap = &cat->iap[i]; + + double oval = -1.0; + for (v = 0; v < hmap_count (&iap->ivmap); ++v) + { + const struct interaction_value *iv = iap->reverse_interaction_value_map[v]; + if (oval == -1.0) + oval = iv->cc; + if (oval != iv->cc) + return false; + } + } + return true; +} + + static void categoricals_dump (const struct categoricals *cat) { @@ -200,8 +242,7 @@ categoricals_dump (const struct categoricals *cat) } printf ("\n"); - - printf ("Number of interactions %d\n", cat->n_iap); + printf ("Number of interactions %zu\n", cat->n_iap); for (i = 0 ; i < cat->n_iap; ++i) { int v; @@ -212,24 +253,32 @@ categoricals_dump (const struct categoricals *cat) ds_init_empty (&str); interaction_to_string (iact, &str); - printf ("\nInteraction: %s (n: %d); ", ds_cstr (&str), iap->n_cats); + printf ("\nInteraction: \"%s\" (number of categories: %d); ", ds_cstr (&str), iap->n_cats); ds_destroy (&str); - printf ("Base subscript: %d\n", iap->base_subscript_short); + printf ("Base index (short/long): %d/%d\n", iap->base_subscript_short, iap->base_subscript_long); printf ("\t("); for (v = 0; v < hmap_count (&iap->ivmap); ++v) { int vv; const struct interaction_value *iv = iap->reverse_interaction_value_map[v]; - + if (v > 0) printf (" "); printf ("{"); for (vv = 0; vv < iact->n_vars; ++vv) { const struct variable *var = iact->vars[vv]; const union value *val = case_data (iv->ccase, var); - - printf ("%g", val->f); + unsigned int varhash = hash_pointer (var, 0); + struct variable_node *vn = lookup_variable (&cat->varmap, var, varhash); + + const int width = var_get_width (var); + unsigned int valhash = value_hash (val, width, 0); + struct value_node *valn = lookup_value (&vn->valmap, val, valhash, width); + + assert (vn->var == var); + + printf ("%.*g(%d)", DBL_DIG + 1, val->f, valn->index); if (vv < iact->n_vars - 1) printf (", "); } @@ -247,23 +296,21 @@ categoricals_destroy (struct categoricals *cat) int i; if (NULL == cat) return; + for (i = 0; i < cat->n_iap; ++i) { struct interaction_value *iv = NULL; /* Interate over each interaction value, and unref any cases that we reffed */ HMAP_FOR_EACH (iv, struct interaction_value, node, &cat->iap[i].ivmap) { -#if 0 - if (cat->payload) - cat->payload->destroy (cat->aux1, iv->user_data); -#endif + if (cat->payload && cat->payload->destroy) + cat->payload->destroy (cat->aux1, cat->aux2, iv->user_data); case_unref (iv->ccase); } free (cat->iap[i].enc_sum); free (cat->iap[i].df_prod); hmap_destroy (&cat->iap[i].ivmap); - interaction_destroy (cat->iap[i].iact); } /* Interate over each variable and delete its value map */ @@ -298,7 +345,7 @@ lookup_case (const struct hmap *map, const struct interaction *iact, const struc return iv; } -bool +bool categoricals_sane (const struct categoricals *cat) { return cat->sane; @@ -310,7 +357,7 @@ categoricals_create (struct interaction *const*inter, size_t n_inter, { size_t i; struct categoricals *cat = xmalloc (sizeof *cat); - + cat->n_iap = n_inter; cat->wv = wv; cat->n_cats_total = 0; @@ -360,7 +407,13 @@ categoricals_update (struct categoricals *cat, const struct ccase *c) { int i; struct variable_node *vn = NULL; - const double weight = cat->wv ? case_data (c, cat->wv)->f : 1.0; + double weight; + + if (NULL == cat) + return; + + weight = cat->wv ? case_data (c, cat->wv)->f : 1.0; + weight = var_force_valid_weight (cat->wv, weight, NULL); assert (NULL == cat->reverse_variable_map_short); assert (NULL == cat->reverse_variable_map_long); @@ -377,13 +430,14 @@ categoricals_update (struct categoricals *cat, const struct ccase *c) if (valn == NULL) { valn = pool_malloc (cat->pool, sizeof *valn); - valn->index = vn->n_vals++; + valn->index = -1; + vn->n_vals++; value_init (&valn->val, width); value_copy (&valn->val, val, width); hmap_insert (&vn->valmap, &valn->node, hash); } - } - + } + for (i = 0 ; i < cat->n_iap; ++i) { const struct interaction *iact = cat->iap[i].iact; @@ -400,13 +454,12 @@ categoricals_update (struct categoricals *cat, const struct ccase *c) if ( NULL == node) { node = pool_malloc (cat->pool, sizeof *node); - node->ccase = case_ref (c); node->cc = weight; hmap_insert (&cat->iap[i].ivmap, &node->node, hash); - if (cat->payload) + if (cat->payload) { node->user_data = cat->payload->create (cat->aux1, cat->aux2); } @@ -419,10 +472,8 @@ categoricals_update (struct categoricals *cat, const struct ccase *c) if (cat->payload) { - double weight = cat->wv ? case_data (c, cat->wv)->f : 1.0; cat->payload->update (cat->aux1, cat->aux2, node->user_data, c, weight); } - } } @@ -455,6 +506,9 @@ categoricals_n_total (const struct categoricals *cat) size_t categoricals_df_total (const struct categoricals *cat) { + if (NULL == cat) + return 0; + return cat->df_sum; } @@ -481,6 +535,10 @@ categoricals_done (const struct categoricals *cat_) int i; int idx_short = 0; int idx_long = 0; + + if (NULL == cat) + return; + cat->df_sum = 0; cat->n_cats_total = 0; @@ -493,26 +551,51 @@ categoricals_done (const struct categoricals *cat_) cat->iap[i].df_prod = iact->n_vars ? xcalloc (iact->n_vars, sizeof (int)) : NULL; cat->iap[i].n_cats = 1; - + for (v = 0 ; v < iact->n_vars; ++v) { + int x; const struct variable *var = iact->vars[v]; struct variable_node *vn = lookup_variable (&cat->varmap, var, hash_pointer (var, 0)); - if (hmap_count (&vn->valmap) == 0) + struct value_node *valnd = NULL; + struct value_node **array ; + + assert (vn->n_vals == hmap_count (&vn->valmap)); + + if (vn->n_vals == 0) { cat->sane = false; return; } - cat->iap[i].df_prod[v] = df * (hmap_count (&vn->valmap) - 1); + /* Sort the VALMAP here */ + array = xcalloc (sizeof *array, vn->n_vals); + x = 0; + HMAP_FOR_EACH (valnd, struct value_node, node, &vn->valmap) + { + /* Note: This loop is probably superfluous, it could be done in the + update stage (at the expense of a realloc) */ + array[x++] = valnd; + } + + sort (array, vn->n_vals, sizeof (*array), + compare_value_node_3way, vn); + + for (x = 0; x < vn->n_vals; ++x) + { + struct value_node *vvv = array[x]; + vvv->index = x; + } + free (array); + + cat->iap[i].df_prod[v] = df * (vn->n_vals - 1); df = cat->iap[i].df_prod[v]; - cat->iap[i].n_cats *= hmap_count (&vn->valmap); + cat->iap[i].n_cats *= vn->n_vals; } - assert (v == iact->n_vars); if (v > 0) cat->df_sum += cat->iap[i].df_prod [v - 1]; @@ -544,7 +627,6 @@ categoricals_done (const struct categoricals *cat_) HMAP_FOR_EACH (ivn, struct interaction_value, node, &iap->ivmap) { iap->reverse_interaction_value_map[x++] = ivn; - } assert (x <= iap->n_cats); @@ -588,7 +670,7 @@ categoricals_done (const struct categoricals *cat_) struct interaction_value *iv = iap->reverse_interaction_value_map[y]; for (x = iap->base_subscript_short; x < iap->base_subscript_short + df ;++x) { - const double bin = categoricals_get_code_for_case (cat, x, iv->ccase); \ + const double bin = categoricals_get_effects_code_for_case (cat, x, iv->ccase); iap->enc_sum [x - iap->base_subscript_short] += bin * iv->cc; } if (cat->payload && cat->payload->calculate) @@ -648,11 +730,13 @@ categoricals_get_sum_by_subscript (const struct categoricals *cat, int subscript return vp->enc_sum[subscript - vp->base_subscript_short]; } + /* Returns unity if the value in case C at SUBSCRIPT is equal to the category for that subscript */ -double +static double categoricals_get_code_for_case (const struct categoricals *cat, int subscript, - const struct ccase *c) + const struct ccase *c, + bool effects_coding) { const struct interaction *iact = categoricals_get_interaction_by_subscript (cat, subscript); @@ -685,14 +769,11 @@ categoricals_get_code_for_case (const struct categoricals *cat, int subscript, const int index = ((subscript - base_index) % iap->df_prod[v] ) / dfp; dfp = iap->df_prod [v]; -#if EFFECTS_CODING - if ( valn->index == df ) + if (effects_coding && valn->index == df ) bin = -1.0; - else -#endif - if ( valn->index != index ) - bin = 0; - + else if ( valn->index != index ) + bin = 0; + result *= bin; } @@ -700,6 +781,28 @@ categoricals_get_code_for_case (const struct categoricals *cat, int subscript, } +/* Returns unity if the value in case C at SUBSCRIPT is equal to the category + for that subscript */ +double +categoricals_get_dummy_code_for_case (const struct categoricals *cat, int subscript, + const struct ccase *c) +{ + return categoricals_get_code_for_case (cat, subscript, c, false); +} + +/* Returns unity if the value in case C at SUBSCRIPT is equal to the category + for that subscript. + Else if it is the last category, return -1. + Otherwise return 0. + */ +double +categoricals_get_effects_code_for_case (const struct categoricals *cat, int subscript, + const struct ccase *c) +{ + return categoricals_get_code_for_case (cat, subscript, c, true); +} + + size_t categoricals_get_n_variables (const struct categoricals *cat) { @@ -708,7 +811,7 @@ categoricals_get_n_variables (const struct categoricals *cat) } -/* Return a case containing the set of values corresponding to +/* Return a case containing the set of values corresponding to the Nth Category of the IACTth interaction */ const struct ccase * categoricals_get_case_by_category_real (const struct categoricals *cat, int iact, int n)