NAME
Ops::Op - Parrot Operation
SYNOPSIS
use Ops::Op;
DESCRIPTION
Ops::Op
represents a Parrot operation (op, for short), as read from an ops file via Ops::OpsFile
, or perhaps even generated by some other means. It is the Perl equivalent of the op_info_t
C struct
defined in include/parrot/op.h.
Op Type
Ops are either auto or manual. Manual ops are responsible for having explicit next-op RETURN()
statements, while auto ops can count on an automatically generated next-op to be appended to the op body.
Note that tools/build/ops2c.pl supplies either 'inline' or 'function' as the op's type, depending on whether the inline
keyword is present in the op definition. This has the effect of causing all ops to be considered manual.
Op Arguments
Note that argument 0 is considered to be the op itself, with arguments 1..9 being the arguments passed to the op.
Op argument direction and type are represented by short one or two letter descriptors.
Op Direction:
i The argument is incoming
o The argument is outgoing
io The argument is both incoming and outgoing
Op Type:
i The argument is an integer register index.
n The argument is a number register index.
p The argument is a PMC register index.
s The argument is a string register index.
ic The argument is an integer constant (in-line).
nc The argument is a number constant index.
pc The argument is a PMC constant index.
sc The argument is a string constant index.
kc The argument is a key constant index.
ki The argument is a key integer register index.
kic The argument is a key integer constant (in-line).
Class Methods
class Ops::Op is PAST::Block;
INIT { pir::load_bytecode("dumper.pbc"); }
new(:$code, :$type, :$name, :@args, :%flags)
Allocates a new bodyless op. A body must be provided eventually for the op to be usable.
$code
is the integer identifier for the op.
$type
is the type of op (see the note on op types above).
$name
is the name of the op.
@args
is a reference to an array of argument type descriptors.
$flags
is a hash reference containing zero or more hints or directives.
Instance Methods
code()
-
Returns the op code.
type()
-
The type of the op, either 'inline' or 'function'.
name()
-
The (short or root) name of the op.
full_name()
-
For argumentless ops, it's the same as
name()
. For ops with arguments, an underscore followed by underscore-separated argument types are appended to the name. func_name()
-
The same as
full_name()
, but with 'Parrot_
' prefixed. experimental()
-
Set or get "experimental" flag for Op.
deprecated()
-
Set or get "deprecated" flag for Op.
method code($code?) { self.attr('code', $code, defined($code)) }
method type($type?) { self.attr('type', $type, defined($type)) }
method name($name?) { self.attr('name', $name, defined($name)) }
method args($args?) { self.attr('args', $args, defined($args)) }
method experimental($args?) { self.attr('experimental', $args, defined($args)) }
method deprecated($args?) { self.attr('deprecated', $args, defined($args)) }
method need_write_barrier() { my $need := 0; # We need write barriers only for (in)out PMC|STR for self.args -> $a { $need := ($a<type> eq 'STR' || $a<type> eq 'PMC') && ($a<direction> eq 'out' || $a<direction> eq 'inout'); return $need if $need; } $need; }
method arg_types($args?) { my $res := self.attr('arg_types', $args, defined($args));
return list() if !defined($res); pir::does__IPS($res, 'array') ?? $res !! list($res); }
method arg_dirs($args?) { self.attr('arg_dirs', $args, defined($args)) }
method arg_type($arg_num) { my @arg_types := self.arg_types; @arg_types[$arg_num]; }
method full_name() { my $name := self.name; my @arg_types := self.arg_types;
#say("# $name arg_types " ~ @arg_types); join('_', $name, |@arg_types); }
method func_name($trans) { return $trans.prefix ~ self.full_name; }
flags()
-
Sets the op's flags. This returns a hash reference, whose keys are any flags (passed as ":flag") specified for the op.
method flags(%flags?) { %flags := self.attr('flags', %flags, defined(%flags)); self.deprecated(%flags<deprecated> ?? 1 !! 0); %flags; }
body($body)
body()
-
Sets/gets the op's code body.
method body() { my $res := ''; for @(self) -> $part { if pir::defined($part) { $res := $res ~ $part<inline>; } } $res; }
jump($jump)
jump()
-
Sets/gets a string containing one or more
op_jump_t
values joined with|
(see include/parrot/op.h). This indicates if and how an op may jump.method jump($jump?) { self.attr('jump', $jump, defined($jump)) }
add_jump($jump)
add_jump($jump)
-
Add a jump flag to this op if it's not there already.
method add_jump($jump) { my $found_jump := 0;
unless self.jump { self.jump(list()) } for self.jump { if $_ eq $jump { $found_jump := 1 } } unless $found_jump { self.jump.push($jump); } }
get_jump()
get_jump()
-
Get the jump flags that apply to this op.
method get_jump() {
if self.jump { return join( '|', |self.jump ); } else { return '0'; } }
source($trans, $op)
-
Returns the
body()
of the op with substitutions made by$trans
(a subclass ofOps::Trans
).method source( $trans ) {
my $prelude := $trans.body_prelude; return $prelude ~ self.get_body( $trans ); }
get_body($trans)
-
Performs the various macro substitutions using the specified transform, correctly handling nested substitutions, and repeating over the whole string until no more substitutions can be made.
VTABLE_
macros are enforced by convertingx->vtable->method
toVTABLE_method
.method get_body( $trans ) {
my %context := hash( trans => $trans, level => 0, ); #work through the op_body tree self.join_children(self, %context); }
# Recursively process body chunks returning string. our multi method to_c(PAST::Val $val, %c) { $val.value; }
our multi method to_c(PAST::Var $var, %c) { if ($var.isdecl) { my $res := $var.vivibase ~ ' ' ~ $var<pointer> ~ ' ' ~ $var.name;
if my $arr := $var<array_size> { $res := $res ~ '[' ~ $arr ~ ']'; } if my $expr := $var.viviself { $res := $res ~ ' = ' ~ self.to_c($expr, %c); } $res; } elsif $var.scope eq 'keyed' { self.to_c($var[0], %c) ~ '[' ~ self.to_c($var[1], %c) ~ ']'; } elsif $var.scope eq 'register' { my $n := +$var.name; %c<trans>.access_arg( self.arg_type($n - 1), $n); } else { # Just ordinary variable $var.name; } }
our %PIROP_MAPPING := hash( :shr('>>'), :shl('<<'),
:shr_assign('>>='), :shl_assign('<<='), :le('<='), :ge('>='), :lt('<'), :gt('>'), :arrow('->'), :dotty('.'), );
our method to_c:pasttype<inline> (PAST::Op $chunk, %c) { return $chunk.inline; }
our method to_c:pasttype<macro> (PAST::Op $chunk, %c) { my $name := $chunk.name; my $children := self.join_children($chunk, %c);
my $trans := %c<trans>; #pir::say('children ' ~ $children); my $ret := Q:PIR< $P0 = find_lex '$trans' $P1 = find_lex '$name' $S0 = $P1 $P1 = find_lex '$children' %r = $P0.$S0($P1) >; #pir::say('RET ' ~ $ret); return $ret; }
our method to_c:pasttype<macro_define> (PAST::Op $chunk, %c) { my @res; @res.push('#define '); #name of macro @res.push($chunk[0]);
@res.push(self.to_c($chunk<macro_args>, %c)) if $chunk<macro_args>; @res.push(self.to_c($chunk<body>, %c)) if $chunk<body>; @res.join(''); }
our method to_c:pasttype<macro_if> (PAST::Op $chunk, %c) { my @res;
@res.push('#if '); # #if isn't parsed semantically yet. @res.push($chunk[0]); #@res.push(self.to_c($trans, $chunk[0])); @res.push("\n"); # 'then' @res.push(self.to_c($chunk[1], %c)); # 'else' @res.push("\n#else\n" ~ self.to_c($chunk[2], %c)) if $chunk[2]; @res.push("\n#endif\n"); @res.join(''); } our method to_c:pasttype<call> (PAST::Op $chunk, %c) { join('', $chunk.name, '(', # Handle args. self.join_children($chunk, %c, ', '), ')', ); }
our method to_c:pasttype<if> (PAST::Op $chunk, %c) { my @res;
if ($chunk<ternary>) { @res.push(self.to_c($chunk[0], %c)); @res.push(" ? "); # 'then' @res.push(self.to_c($chunk[1], %c)); # 'else' @res.push(" : "); @res.push(self.to_c($chunk[2], %c)); } else { @res.push('if ('); @res.push(self.to_c($chunk[0], %c)); @res.push(") "); # 'then' # single statement. Make it pretty. @res.push(self.to_c($chunk[1], %c)); # 'else' if $chunk[2] { @res.push("\n"); @res.push(indent(%c)); @res.push("else "); @res.push(self.to_c($chunk[2], %c)); } } @res.join(''); }
our method to_c:pasttype<while> (PAST::Op $chunk, %c) { join('', 'while (', self.to_c($chunk[0], %c), ') ', self.to_c($chunk[1], %c), ); }
our method to_c:pasttype<do-while> (PAST::Op $chunk, %c) { join('', 'do ', self.to_c($chunk[0], %c), ' while (', self.to_c($chunk[1], %c), ');', ); }
our method to_c:pasttype<for> (PAST::Op $chunk, %c) { join('', 'for (', $chunk[0] ?? self.to_c($chunk[0], %c) !! '', '; ', $chunk[1] ?? self.to_c($chunk[1], %c) !! '', '; ', $chunk[2] ?? self.to_c($chunk[2], %c) !! '', ') ', self.to_c($chunk[3], %c), ); }
our method to_c:pasttype<switch> (PAST::Op $chunk, %c) { join('', 'switch (', self.to_c($chunk[0], %c), ') {', "\n", self.to_c($chunk[1], %c), "\n", indent(%c), "}", ); }
our method to_c:pasttype<undef> (PAST::Op $chunk, %c) { my $pirop := $chunk.pirop;
if $pirop { # Some infix stuff if $pirop eq ',' { self.join_children($chunk, %c, ', '); } elsif $pirop eq '=' { self.to_c($chunk[0], %c) ~ ' = ' ~ self.to_c($chunk[1], %c) } elsif ($pirop eq 'arrow') || ($pirop eq 'dotty') { self.to_c($chunk[0], %c) ~ %PIROP_MAPPING{$pirop} ~ self.to_c($chunk[1], %c) } elsif $chunk.name ~~ / infix / { '(' ~ self.to_c($chunk[0], %c) ~ ' ' ~ (%PIROP_MAPPING{$pirop} // $pirop) ~ ' ' ~ self.to_c($chunk[1], %c) ~ ')'; } elsif $chunk.name ~~ / prefix / { '(' ~ (%PIROP_MAPPING{$pirop} // $pirop) ~ self.to_c($chunk[0], %c) ~ ')'; } elsif $chunk.name ~~ / postfix / { '(' ~ self.to_c($chunk[0], %c) ~ (%PIROP_MAPPING{$pirop} // $pirop) ~ ')'; } else { _dumper($chunk); pir::die("Unhandled chunk for pirop"); } } elsif $chunk.returns { # Handle "cast" join('', '(', $chunk.returns, ')', self.to_c($chunk[0], %c), ); } elsif $chunk<control> { $chunk<control>; } elsif $chunk<label> { # Do nothing. Empty label for statement. ""; } else { _dumper($chunk); pir::die("Unhandled chunk"); } }
our multi method to_c(PAST::Op $chunk, %c) { my @res;
@res.push($chunk<label> ~ "\n" ~ indent(%c)) if $chunk<label>; my $type := $chunk.pasttype // 'undef'; my $sub := pir::find_sub_not_null__ps('to_c:pasttype<' ~ $type ~ '>'); @res.push('(') if $chunk<wrap>; @res.push($sub(self, $chunk, %c)); @res.push(')') if $chunk<wrap>; @res.join(''); }
our multi method to_c(PAST::Stmts $chunk, %c) { %c<level>++ unless $chunk[0] ~~ PAST::Block;
my @res; for @($chunk) { @res.push(indent($_, %c)) unless $_ ~~ PAST::Block; @res.push(self.to_c($_, %c)); @res.push(";") if need_semicolon($_); @res.push("\n"); } %c<level>-- unless $chunk[0] ~~ PAST::Block; @res.join(''); }
our multi method to_c(PAST::Block $chunk, %c) { # Put newline after variable declarations. my $need_space := need_space($chunk[0]);
my @res; @res.push(indent($chunk, %c) ~ $chunk<label> ~ "\n" ~ indent(%c)) if $chunk<label>; %c<level>++; @res.push("\{\n"); for @($chunk) { if $need_space && !need_space($_) { # Hack. If this $chunk doesn't need semicolon it will put newline before @res.push("\n"); $need_space := 0; } @res.push(indent($_, %c)); @res.push(self.to_c($_, %c)); @res.push(need_semicolon($_) ?? ";" !! "\n"); @res.push("\n"); } %c<level>--; @res.push(indent(%c)); @res.push("}"); @res.join(''); }
sub need_space($past) { ($past ~~ PAST::Var) && $past.isdecl; }
sub need_semicolon($past) { return 0 if $past ~~ PAST::Block; return 1 unless $past ~~ PAST::Op;
my $pasttype := $past.pasttype; return 1 unless $pasttype; return 0 if $pasttype eq 'if'; return 0 if $pasttype eq 'for'; return 0 if $pasttype eq 'while'; return 0 if $pasttype eq 'do-while'; return 0 if $pasttype eq 'switch'; return 1; }
# Stub! our multi method to_c(String $str, %c) { $str; }
size()
-
Returns the op's number of arguments. Note that this also includes the op itself as one argument.
method size() { return pir::does__IPs(self.args, 'array') ?? +self.args + 1 !! 2; }
method join_children (PAST::Node $node, %c, $joiner?) { @($node).map(-> $_ { self.to_c($_, %c) }).join($joiner // ''); }
our multi sub indent($chunk, %c) { pir::repeat(' ', %c<level> * 4 - ($chunk<label> ?? 2 !! 0)); }
our multi sub indent(%c) { pir::repeat(' ', %c<level> * 4); }
SEE ALSO
HISTORY
Author: Gregor N. Purdy <gregor@focusresearch.com>
Migrate to NQP: Vasily Chekalkin <bacek@bacek.com>
1;
# Local Variables: # mode: cperl # cperl-indent-level: 4 # fill-column: 100 # End: # vim: ft=perl6 expandtab shiftwidth=4:
23 POD Errors
The following errors were encountered while parsing the POD:
- Around line 4:
=begin without a target?
- Around line 62:
You can't have =items (as at line 74) unless the first thing after the =over is an =item
- Around line 64:
'=end' without a target?
- Around line 72:
=begin without a target?
- Around line 127:
'=end' without a target?
- Around line 179:
=begin without a target?
- Around line 186:
'=end' without a target?
- Around line 194:
=begin without a target?
- Around line 202:
'=end' without a target?
- Around line 214:
=begin without a target?
- Around line 224:
'=end' without a target?
- Around line 228:
=begin without a target?
- Around line 236:
'=end' without a target?
- Around line 252:
=begin without a target?
- Around line 260:
'=end' without a target?
- Around line 272:
=begin without a target?
- Around line 279:
'=end' without a target?
- Around line 287:
=begin without a target?
- Around line 298:
'=end' without a target?
- Around line 652:
=begin without a target?
- Around line 659:
'=end' without a target?
- Around line 678:
=begin without a target?
- Around line 700:
'=end' without a target?