summaryrefslogtreecommitdiff
path: root/src/fjbg/ch/epfl/lamp/fjbg/JCodeAttribute.java
blob: 76dda2fe7627c1d263409cb645581c71d00fdde5 (plain) (blame)
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
/* FJBG -- Fast Java Bytecode Generator
 * Copyright 2002-2012 LAMP/EPFL
 * @author  Michel Schinz
 */

package ch.epfl.lamp.fjbg;

import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.util.Iterator;
import java.util.List;

/**
 * Code attribute, containing code of methods.
 *
 * A Code attribute contains the JVM instructions and auxiliary information
 * for a single method, instance initialization method, or class or interface
 * initialization method. See section 4.8.3 of the JVM specification.
 *
 * @author Michel Schinz, Stephane Micheloud
 * @version 1.1
 */

public class JCodeAttribute extends JAttribute {
    protected final JCode code;
    protected final JMethod owner;
    protected static int UNKNOWN_STACK_SIZE = Integer.MIN_VALUE;
    protected final int maxStackSize;
    protected final int maxLocals;

    public JCodeAttribute(FJBGContext context, JClass clazz, JMethod owner) {
        super(context, clazz);
        this.owner = owner;

        this.maxStackSize = UNKNOWN_STACK_SIZE;
        this.maxLocals = 0; // unknown
        this.code = owner.getCode();

        assert clazz == owner.getOwner();
    }

    public JCodeAttribute(FJBGContext context,
                          JClass clazz,
                          Object owner,
                          String name,
                          int size,
                          DataInputStream stream)
        throws IOException {
        super(context, clazz, name);
        this.owner = (JMethod)owner;

        this.maxStackSize = stream.readShort();
        this.maxLocals = stream.readShort();
        this.code = context.JCode(clazz, (JMethod)owner, stream);

        int handlersCount = stream.readShort();
        for (int i = 0; i < handlersCount; ++i)
            code.addExceptionHandler(code.new ExceptionHandler(stream));
        List/*<JAttribute>*/ attributes =
            JAttribute.readFrom(context, clazz, code, stream);
        Iterator attrIt = attributes.iterator();
        while (attrIt.hasNext())
            code.addAttribute((JAttribute)attrIt.next());

        assert name.equals(getName());
    }

    public String getName() { return "Code"; }

    // Follows javap output format for Code attribute.
    /*@Override*/ public String toString() {
        StringBuffer buf = new StringBuffer("  Code:");
        buf.append("\n   Stack=");
        buf.append(maxStackSize);
        buf.append(", Locals=");
        buf.append(maxLocals);
        buf.append(", Args_size=");
        buf.append(owner.getArgsSize());
        buf.append(code);
        buf.append("\n");
        Iterator it = code.getAttributes().iterator();
        while (it.hasNext()) {
            JAttribute attr = (JAttribute)it.next();
            buf.append(attr);
            buf.append("\n");
        }
        return buf.toString();
    }

    protected int getSize() {
        int handlersNum = code.getExceptionHandlers().size();

        int attrsSize = 0;
        Iterator attrsIt = code.getAttributes().iterator();
        while (attrsIt.hasNext()) {
            JAttribute attr = (JAttribute)attrsIt.next();
            attrsSize += attr.getSize() + 6;
        }

        return 2                // max stack
            + 2                 // max locals
            + 4                 // code size
            + code.getSize()    // code
            + 2                 // exception table size
            + 8 * handlersNum   // exception table
            + 2                 // attributes count
            + attrsSize;        // attributes
    }

    protected void writeContentsTo(DataOutputStream stream) throws IOException {
        List/*<ExceptionHandler>*/ handlers = code.getExceptionHandlers();

        stream.writeShort(code.getMaxStackSize());
        stream.writeShort(owner.getMaxLocals());

        code.writeTo(stream);

        stream.writeShort(handlers.size());
        Iterator handlerIt = handlers.iterator();
        while (handlerIt.hasNext())
            ((JCode.ExceptionHandler)handlerIt.next()).writeTo(stream);
        JAttribute.writeTo(code.getAttributes(), stream);
    }
}