view cbctools/change_OP_to_cbc.pl @ 34:0853778b49ee

fix insert cbc_next(i) for throw code sgemtns
author Takahiro SHIMIZU <anatofuz@cr.ie.u-ryukyu.ac.jp>
date Tue, 13 Nov 2018 17:54:28 +0900
parents a5f4b3f1c5d1
children 29dd3807070e
line wrap: on
line source

#!/usr/bin/env perl
use strict;
use warnings;

my $cbc_file = shift or die; # src/core/cbc-interp.cbc

open my $fh, '<',$cbc_file;
my @cbc_lines= <$fh>;
close $fh;

my @rewritec = ();
my $indent = " " x 12;
my $none_left_blanket = 0;

my $i = 0;

my @upside = ();
for (;$i < scalar(@cbc_lines); $i++){

    $cbc_lines[$i] = change_i($cbc_lines[$i]);

    #$cbc_lines[$i] =~ s/reg_base/i->reg_base/g;
    #$cbc_lines[$i] =~ s/tc/i->tc/g;
    #$cbc_lines[$i] =~ s/tc/i->tc/g;

    $cbc_lines[$i] =~ s/GET_(REG|LEX)\((.*?)\)/GET_$1($2,i)/g;

    push @upside,$cbc_lines[$i];
    if ($cbc_lines[$i+1] =~ /\/\* This is the interpreter run loop. We have one of these per thread. \*\//){
        last;
    }
}

my @middle = ();

for (;$i < scalar(@cbc_lines); $i++){
    push @middle,$cbc_lines[$i];
    if ($cbc_lines[$i] =~ /DISPATCH\(NEXT_OP\)/){ # DISPATCHの中身を書き換えるのでそこまで飛ばす
        $i++;
        last;
    }
}
my @dispatch = ();

for (;$i < scalar(@cbc_lines); $i++){
    # check OP(.*) {  codes

    if ($cbc_lines[$i] =~ /#if MVM_CGOTO/){
        $i--;
        last;
    }

    if ($cbc_lines[$i] =~ /^\s+OP\((.*)\):/ ){
        my $opcode = $1;
        #push @dispatch,"${indent}OP($opcode):\n";
        $none_left_blanket = $cbc_lines[$i] =~ /{/ ? 0 : 1;
        # transrate OP(HOGE) to __code hoge(INTERP *i){
        $cbc_lines[$i]  =  "__code "."cbc_$opcode"."(INTERP i){\n";
        #$cbc_lines[$i]  = $indent. "__code ".$opcode."(INTERP i){\n";
        push @rewritec,$cbc_lines[$i];

        # 次の行に移動
        $i++;

        # この行がOP()だった場合
        #  OP(DEPRECATED_4):
        #  OP(DEPRECATED_5): <- $i
        # このような宣言になっているので$iにgotoするように関数を書き直し
        # $iの部分の関数定義を次ループでするために一行戻して再ループ

        if ($cbc_lines[$i] =~ /^\s+OP\((.*)\):/){
            push @rewritec,"    goto cbc_$1(i);\n";
            #push @rewritec,"$indent   goto $1(i);\n";
            insert_right_blanket();
            $i--;
            next;
        }

        # 例外だったらかえってこないはずなのでgoto
        if ( $cbc_lines[$i] =~ /MVM_exception_throw_adhoc/ && $cbc_lines[$i+1] =~ /OP\(/){
            push @rewritec, change_i($cbc_lines[$i]);
            insert_cbc_next();
            insert_right_blanket();
            next;
        }
    }

    # 普通の行は変換してinsertする
    if ($cbc_lines[$i] =~ /}/){
        if ($cbc_lines[$i+1] =~ /OP/){
            $cbc_lines[$i] = "}\n";
        }
    }
    push @rewritec,change_i($cbc_lines[$i]);


   if ($i != scalar(@cbc_lines)-1 && $cbc_lines[$i+1] =~ /OP/ && $none_left_blanket){
        insert_right_blanket();
        $none_left_blanket = 0;
    }
}

my @after = ();
map { push @after,$cbc_lines[$_]} ($i+1.. scalar(@cbc_lines)-1);
#map { print; } (@upside,@rewritec,@middle,@dispatch,@after);
map { print; } (@upside,@rewritec,@middle,@after);


sub change_i {
    my $str = shift;
    $str =~ s/^op/i->op/g;
    $str =~ s/cur_op/i->cur_op/g;
    $str =~ s/tc/i->tc/g;
    $str =~ s/cur_callsite/i->cur_callsite/g;
    $str =~ s/NEXT;/cbc_next(i);/;
    $str =~ s/                /    /g;
    return $str;
}

sub insert_cbc_next{
    #push @rewritec,"$indent}\n";
    push @rewritec,"    goto cbc_next(i);\n";
}

sub insert_right_blanket {
    #push @rewritec,"$indent}\n";
    push @rewritec,"}\n";
}