File: class.c

package info (click to toggle)
boost-jam 3.1.10-1
  • links: PTS
  • area: main
  • in suites: sarge
  • size: 1,004 kB
  • ctags: 1,703
  • sloc: ansic: 11,378; yacc: 456; sh: 192; makefile: 43
file content (128 lines) | stat: -rw-r--r-- 3,282 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128

#include "class.h"
#include "strings.h"
#include "variable.h"
#include "frames.h"
#include "rules.h"
#include "newstr.h"

#include "hash.h"

static struct hash* classes = 0;

static void check_defined(LIST* class_names)
{
    for (; class_names; class_names = class_names->next) {
        char** p = &class_names->string;
        if (!hashcheck(classes, (HASHDATA**)&p)) {
            printf("Class %s is not defined\n", class_names->string);
            abort();
        }
    }
}

static char* class_module_name(char* declared_name)
{
    string name[1];
    char* result;
    
    string_new(name);
    string_append(name, "class@");
    string_append(name, declared_name);

    result = newstr(name->value);
    string_free(name);
    
    return result;
}

struct import_base_data {
    char* base_name;
    module_t* base_module;
    module_t* class_module;
};

static void import_base_rule(void* r_, void* d_)
{
    RULE* r = (RULE*)r_;
    RULE* ir1;
    RULE* ir2;
    struct import_base_data* d = (struct import_base_data*)d_;
    string qualified_name[1];
    int basename_lenght = strlen(d->base_name)+1;

    string_new(qualified_name);
    string_append(qualified_name, d->base_name);
    string_push_back(qualified_name, '.');
    string_append(qualified_name, r->name);    

    ir1 = import_rule(r, d->class_module, r->name);
    ir2 = import_rule(r, d->class_module, qualified_name->value);

    /* Copy 'exported' flag. */
    ir1->exported = ir2->exported = r->exported;

    /* If we're importing class method, localize it. */
    if (r->module == d->base_module 
        || r->module->class_module && r->module->class_module == d->base_module) {
        ir1->module = ir2->module = d->class_module;        
    }
        
    string_free(qualified_name);
}

/** For each exported rule 'n', declared in class module for base,
    imports that rule in 'class' as 'n' and as 'base.n'. Imported
    rules are localized and marked as exported.
*/
static void import_base_rules(module_t* class, char* base)
{
    module_t* base_module = bindmodule(class_module_name(base));
    struct import_base_data d;
    d.base_name = base;
    d.base_module = base_module;
    d.class_module = class;

    if (base_module->rules)
        hashenumerate(base_module->rules, import_base_rule, &d);

    import_module( imported_modules(base_module), class );
}

char* make_class_module(LIST* xname, LIST* bases, FRAME* frame)
{
    char* name = class_module_name(xname->string);
    char** pp = &xname->string;
    module_t* class_module = 0;
    module_t* outer_module = frame->module;    

    if (!classes)
        classes = hashinit(sizeof(char*), "classes");

    
    if (hashcheck(classes, (HASHDATA**)&pp)) {        
        printf("Class %s already defined\n", xname->string);
        abort();
    } else {
        hashenter(classes, (HASHDATA**)&pp);
    }
    check_defined(bases);
    
    class_module = bindmodule(name);

    exit_module( outer_module );
    enter_module( class_module );

    var_set("__name__", xname, VAR_SET);
    var_set("__bases__", bases, VAR_SET);
    
    exit_module( class_module );
    enter_module( outer_module );
    
    for(; bases; bases = bases->next)
        import_base_rules(class_module, bases->string);



    return name;
}