/
Decl.pm
150 lines (123 loc) · 3.71 KB
/
Decl.pm
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
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
use strict;
use warnings;
use 5.010;
{
package Decl;
use Moose;
sub do_preinit {}
sub do_enter {}
sub write {}
__PACKAGE__->meta->make_immutable;
no Moose;
}
{
package Decl::PreInit;
use Moose;
extends 'Decl';
has var => (isa => 'Str', is => 'ro', predicate => 'has_var');
has code => (isa => 'Body', is => 'ro', required => 1);
has shared => (isa => 'Bool', is => 'ro', default => 0);
sub do_preinit {
my ($self, $cg, $body) = @_;
$self->code->outer($body);
$cg->open_protopad;
$self->code->do_preinit($cg);
$cg->close_sub($self->code->code);
$cg->call_sub($self->has_var, 0);
$cg->proto_var($self->var) if $self->has_var;
}
sub do_enter {
my ($self, $cg, $body) = @_;
return unless $self->has_var;
if ($self->shared) {
$cg->share_lex($self->var);
} else {
$cg->copy_lex($self->var);
}
}
sub write {
my ($self, $body) = @_;
$self->code->outer($body);
$self->code->write;
}
__PACKAGE__->meta->make_immutable;
no Moose;
}
{
package Decl::Sub;
use Moose;
extends 'Decl';
has var => (isa => 'Str', is => 'ro', required => 1);
has code => (isa => 'Body', is => 'ro', required => 1);
sub do_preinit {
my ($self, $cg, $body) = @_;
$self->code->outer($body);
$cg->open_protopad;
$self->code->do_preinit($cg);
$cg->close_sub($self->code->code);
$cg->clr_call_direct('Kernel.NewROVar', 1);
$cg->proto_var($self->var);
}
sub do_enter {
my ($self, $cg, $body) = @_;
$cg->clone_lex($self->var);
}
sub write {
my ($self, $body) = @_;
$self->code->outer($body);
$self->code->write;
}
__PACKAGE__->meta->make_immutable;
no Moose;
}
{
package Decl::Class;
use Moose;
has name => (is => 'ro', isa => 'Str', predicate => 'has_name');
has var => (is => 'ro', isa => 'Str', required => 1);
has stub => (is => 'ro', isa => 'Bool', default => 1);
has parents => (is => 'ro', isa => 'ArrayRef', default => sub { [] });
# the body is a very sublike thing; it has a preinit existance, and a
# lexical scope. but instead of just a Sub, it constructs a ClassHOW at
# preinit
has body => (is => 'ro', isa => 'Body::Class', required => 1);
sub do_preinit {
my ($self, $cg, $body) = @_;
$cg->scopelexget("ClassHOW", $body);
$cg->dup_fetch;
$cg->string_var($self->name // 'ANON');
$cg->call_method(1, "new", 1);
$cg->dup;
$cg->push_aux('how');
$cg->proto_var($self->var . '!HOW');
# TODO: Initialize the protoobject to a failure here so an awesome error
# is produced if someone tries to use an incomplete class in a BEGIN.
$cg->push_null('Variable');
$cg->proto_var($self->var);
$self->body->outer($body);
$self->body->var($self->var);
$cg->open_protopad;
$cg->peek_aux('how');
$cg->dup_fetch;
$cg->peek_aux('protopad');
$cg->clr_call_direct('Kernel.NewROVar', 1);
$cg->call_method(0, "push-scope", 1);
$self->body->do_preinit($cg);
$cg->close_sub($self->body->code);
$cg->proto_var($self->var . '!BODY');
}
sub do_enter {
my ($self, $cg, $body) = @_;
$cg->share_lex($self->var . '!HOW');
$cg->clone_lex($self->var . '!BODY');
}
sub write {
my ($self, $body) = @_;
$self->body->var($self->var);
$self->body->outer($body);
$self->body->write;
}
__PACKAGE__->meta->make_immutable;
no Moose;
}
1;