mirror of https://github.com/ethereum/go-ethereum
commit
50fdfb127a
@ -0,0 +1,264 @@ |
|||||||
|
import QtQuick 2.0 |
||||||
|
import QtQuick.Controls 1.0; |
||||||
|
import QtQuick.Layouts 1.0; |
||||||
|
import QtQuick.Dialogs 1.0; |
||||||
|
import QtQuick.Window 2.1; |
||||||
|
import QtQuick.Controls.Styles 1.1 |
||||||
|
import Ethereum 1.0 |
||||||
|
|
||||||
|
ApplicationWindow { |
||||||
|
visible: false |
||||||
|
title: "IceCREAM" |
||||||
|
minimumWidth: 1280 |
||||||
|
minimumHeight: 900 |
||||||
|
width: 1290 |
||||||
|
height: 900 |
||||||
|
|
||||||
|
property alias codeText: codeEditor.text |
||||||
|
property alias dataText: rawDataField.text |
||||||
|
|
||||||
|
MenuBar { |
||||||
|
Menu { |
||||||
|
title: "Debugger" |
||||||
|
MenuItem { |
||||||
|
text: "Run" |
||||||
|
shortcut: "Ctrl+r" |
||||||
|
onTriggered: debugCurrent() |
||||||
|
} |
||||||
|
|
||||||
|
MenuItem { |
||||||
|
text: "Next" |
||||||
|
shortcut: "Ctrl+n" |
||||||
|
onTriggered: dbg.next() |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
SplitView { |
||||||
|
anchors.fill: parent |
||||||
|
property var asmModel: ListModel { |
||||||
|
id: asmModel |
||||||
|
} |
||||||
|
TableView { |
||||||
|
id: asmTableView |
||||||
|
width: 200 |
||||||
|
TableViewColumn{ role: "value" ; title: "" ; width: 100 } |
||||||
|
model: asmModel |
||||||
|
} |
||||||
|
|
||||||
|
Rectangle { |
||||||
|
color: "#00000000" |
||||||
|
anchors.left: asmTableView.right |
||||||
|
anchors.right: parent.right |
||||||
|
SplitView { |
||||||
|
orientation: Qt.Vertical |
||||||
|
anchors.fill: parent |
||||||
|
|
||||||
|
Rectangle { |
||||||
|
color: "#00000000" |
||||||
|
height: 500 |
||||||
|
anchors.left: parent.left |
||||||
|
anchors.right: parent.right |
||||||
|
|
||||||
|
TextArea { |
||||||
|
id: codeEditor |
||||||
|
anchors.top: parent.top |
||||||
|
anchors.bottom: parent.bottom |
||||||
|
anchors.left: parent.left |
||||||
|
anchors.right: settings.left |
||||||
|
} |
||||||
|
|
||||||
|
Column { |
||||||
|
id: settings |
||||||
|
spacing: 5 |
||||||
|
width: 300 |
||||||
|
height: parent.height |
||||||
|
anchors.right: parent.right |
||||||
|
anchors.top: parent.top |
||||||
|
anchors.bottom: parent.bottom |
||||||
|
|
||||||
|
Label { |
||||||
|
text: "Arbitrary data" |
||||||
|
} |
||||||
|
TextArea { |
||||||
|
id: rawDataField |
||||||
|
anchors.left: parent.left |
||||||
|
anchors.right: parent.right |
||||||
|
height: 150 |
||||||
|
} |
||||||
|
|
||||||
|
Label { |
||||||
|
text: "Amount" |
||||||
|
} |
||||||
|
TextField { |
||||||
|
id: txValue |
||||||
|
width: 200 |
||||||
|
placeholderText: "Amount" |
||||||
|
validator: RegExpValidator { regExp: /\d*/ } |
||||||
|
} |
||||||
|
Label { |
||||||
|
text: "Amount of gas" |
||||||
|
} |
||||||
|
TextField { |
||||||
|
id: txGas |
||||||
|
width: 200 |
||||||
|
validator: RegExpValidator { regExp: /\d*/ } |
||||||
|
text: "10000" |
||||||
|
placeholderText: "Gas" |
||||||
|
} |
||||||
|
Label { |
||||||
|
text: "Gas price" |
||||||
|
} |
||||||
|
TextField { |
||||||
|
id: txGasPrice |
||||||
|
width: 200 |
||||||
|
placeholderText: "Gas price" |
||||||
|
text: "1000000000000" |
||||||
|
validator: RegExpValidator { regExp: /\d*/ } |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
SplitView { |
||||||
|
orientation: Qt.Vertical |
||||||
|
id: inspectorPane |
||||||
|
height: 500 |
||||||
|
|
||||||
|
SplitView { |
||||||
|
orientation: Qt.Horizontal |
||||||
|
height: 150 |
||||||
|
|
||||||
|
TableView { |
||||||
|
id: stackTableView |
||||||
|
property var stackModel: ListModel { |
||||||
|
id: stackModel |
||||||
|
} |
||||||
|
height: parent.height |
||||||
|
width: 300 |
||||||
|
TableViewColumn{ role: "value" ; title: "Stack" ; width: 200 } |
||||||
|
model: stackModel |
||||||
|
} |
||||||
|
|
||||||
|
TableView { |
||||||
|
id: memoryTableView |
||||||
|
property var memModel: ListModel { |
||||||
|
id: memModel |
||||||
|
} |
||||||
|
height: parent.height |
||||||
|
width: parent.width - stackTableView.width |
||||||
|
TableViewColumn{ id:mnumColmn ; role: "num" ; title: "#" ; width: 50} |
||||||
|
TableViewColumn{ role: "value" ; title: "Memory" ; width: 750} |
||||||
|
model: memModel |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
Rectangle { |
||||||
|
height: 100 |
||||||
|
width: parent.width |
||||||
|
TableView { |
||||||
|
id: storageTableView |
||||||
|
property var memModel: ListModel { |
||||||
|
id: storageModel |
||||||
|
} |
||||||
|
height: parent.height |
||||||
|
width: parent.width |
||||||
|
TableViewColumn{ id: key ; role: "key" ; title: "#" ; width: storageTableView.width / 2} |
||||||
|
TableViewColumn{ role: "value" ; title: "Storage" ; width: storageTableView.width / 2} |
||||||
|
model: storageModel |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
Rectangle { |
||||||
|
height: 200 |
||||||
|
width: parent.width |
||||||
|
TableView { |
||||||
|
id: logTableView |
||||||
|
property var logModel: ListModel { |
||||||
|
id: logModel |
||||||
|
} |
||||||
|
height: parent.height |
||||||
|
width: parent.width |
||||||
|
TableViewColumn{ id: message ; role: "message" ; title: "log" ; width: logTableView.width } |
||||||
|
model: logModel |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
toolBar: ToolBar { |
||||||
|
RowLayout { |
||||||
|
spacing: 5 |
||||||
|
|
||||||
|
Button { |
||||||
|
property var enabled: true |
||||||
|
id: debugStart |
||||||
|
onClicked: { |
||||||
|
debugCurrent() |
||||||
|
} |
||||||
|
text: "Debug" |
||||||
|
} |
||||||
|
|
||||||
|
Button { |
||||||
|
property var enabled: true |
||||||
|
id: debugNextButton |
||||||
|
onClicked: { |
||||||
|
dbg.next() |
||||||
|
} |
||||||
|
text: "Next" |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
function debugCurrent() { |
||||||
|
dbg.debug(txValue.text, txGas.text, txGasPrice.text, codeEditor.text, rawDataField.text) |
||||||
|
} |
||||||
|
|
||||||
|
function setAsm(asm) { |
||||||
|
asmModel.append({asm: asm}) |
||||||
|
} |
||||||
|
|
||||||
|
function clearAsm() { |
||||||
|
asmModel.clear() |
||||||
|
} |
||||||
|
|
||||||
|
function setInstruction(num) { |
||||||
|
//asmTableView.selection.clear() |
||||||
|
//asmTableView.selection.select(num) |
||||||
|
} |
||||||
|
|
||||||
|
function setMem(mem) { |
||||||
|
memModel.append({num: mem.num, value: mem.value}) |
||||||
|
} |
||||||
|
function clearMem(){ |
||||||
|
memModel.clear() |
||||||
|
} |
||||||
|
|
||||||
|
function setStack(stack) { |
||||||
|
stackModel.append({value: stack}) |
||||||
|
} |
||||||
|
function addDebugMessage(message){ |
||||||
|
debuggerLog.append({value: message}) |
||||||
|
} |
||||||
|
|
||||||
|
function clearStack() { |
||||||
|
stackModel.clear() |
||||||
|
} |
||||||
|
|
||||||
|
function clearStorage() { |
||||||
|
storageModel.clear() |
||||||
|
} |
||||||
|
|
||||||
|
function setStorage(storage) { |
||||||
|
storageModel.append({key: storage.key, value: storage.value}) |
||||||
|
} |
||||||
|
|
||||||
|
function setLog(msg) { |
||||||
|
logModel.insert(0, {message: msg}) |
||||||
|
} |
||||||
|
|
||||||
|
function clearLog() { |
||||||
|
logModel.clear() |
||||||
|
} |
||||||
|
} |
After Width: | Height: | Size: 4.2 KiB |
@ -1,196 +0,0 @@ |
|||||||
import QtQuick 2.0 |
|
||||||
import QtQuick.Controls 1.0; |
|
||||||
import QtQuick.Layouts 1.0; |
|
||||||
import QtQuick.Dialogs 1.0; |
|
||||||
import QtQuick.Window 2.1; |
|
||||||
import QtQuick.Controls.Styles 1.1 |
|
||||||
import Ethereum 1.0 |
|
||||||
|
|
||||||
Component { |
|
||||||
id: newContract |
|
||||||
Column { |
|
||||||
id: mainContractColumn |
|
||||||
function contractFormReady(){ |
|
||||||
if(codeView.text.length > 0 && txValue.text.length > 0 && txGas.text.length > 0 && txGasPrice.length > 0) { |
|
||||||
txButton.state = "READY" |
|
||||||
}else{ |
|
||||||
txButton.state = "NOTREADY" |
|
||||||
} |
|
||||||
} |
|
||||||
states: [ |
|
||||||
State{ |
|
||||||
name: "ERROR" |
|
||||||
PropertyChanges { target: txResult; visible:true} |
|
||||||
PropertyChanges { target: codeView; visible:true} |
|
||||||
}, |
|
||||||
State { |
|
||||||
name: "DONE" |
|
||||||
PropertyChanges { target: txValue; visible:false} |
|
||||||
PropertyChanges { target: txGas; visible:false} |
|
||||||
PropertyChanges { target: txGasPrice; visible:false} |
|
||||||
PropertyChanges { target: codeView; visible:false} |
|
||||||
PropertyChanges { target: txButton; visible:false} |
|
||||||
PropertyChanges { target: txDataLabel; visible:false} |
|
||||||
|
|
||||||
PropertyChanges { target: txResult; visible:true} |
|
||||||
PropertyChanges { target: txOutput; visible:true} |
|
||||||
PropertyChanges { target: newTxButton; visible:true} |
|
||||||
}, |
|
||||||
State { |
|
||||||
name: "SETUP" |
|
||||||
PropertyChanges { target: txValue; visible:true; text: ""} |
|
||||||
PropertyChanges { target: txGas; visible:true; text: ""} |
|
||||||
PropertyChanges { target: txGasPrice; visible:true; text: ""} |
|
||||||
PropertyChanges { target: codeView; visible:true; text: ""} |
|
||||||
PropertyChanges { target: txButton; visible:true} |
|
||||||
PropertyChanges { target: txDataLabel; visible:true} |
|
||||||
|
|
||||||
PropertyChanges { target: txResult; visible:false} |
|
||||||
PropertyChanges { target: txOutput; visible:false} |
|
||||||
PropertyChanges { target: newTxButton; visible:false} |
|
||||||
} |
|
||||||
] |
|
||||||
width: 400 |
|
||||||
spacing: 5 |
|
||||||
anchors.left: parent.left |
|
||||||
anchors.top: parent.top |
|
||||||
anchors.leftMargin: 5 |
|
||||||
anchors.topMargin: 5 |
|
||||||
|
|
||||||
TextField { |
|
||||||
id: txValue |
|
||||||
width: 200 |
|
||||||
placeholderText: "Amount" |
|
||||||
validator: RegExpValidator { regExp: /\d*/ } |
|
||||||
onTextChanged: { |
|
||||||
contractFormReady() |
|
||||||
} |
|
||||||
} |
|
||||||
TextField { |
|
||||||
id: txGas |
|
||||||
width: 200 |
|
||||||
validator: RegExpValidator { regExp: /\d*/ } |
|
||||||
placeholderText: "Gas" |
|
||||||
onTextChanged: { |
|
||||||
contractFormReady() |
|
||||||
} |
|
||||||
} |
|
||||||
TextField { |
|
||||||
id: txGasPrice |
|
||||||
width: 200 |
|
||||||
placeholderText: "Gas price" |
|
||||||
validator: RegExpValidator { regExp: /\d*/ } |
|
||||||
onTextChanged: { |
|
||||||
contractFormReady() |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
Row { |
|
||||||
id: rowContract |
|
||||||
ExclusiveGroup { id: contractTypeGroup } |
|
||||||
RadioButton { |
|
||||||
id: createContractRadio |
|
||||||
text: "Create contract" |
|
||||||
checked: true |
|
||||||
exclusiveGroup: contractTypeGroup |
|
||||||
onClicked: { |
|
||||||
txFuelRecipient.visible = false |
|
||||||
txDataLabel.text = "Contract code" |
|
||||||
} |
|
||||||
} |
|
||||||
RadioButton { |
|
||||||
id: runContractRadio |
|
||||||
text: "Run contract" |
|
||||||
exclusiveGroup: contractTypeGroup |
|
||||||
onClicked: { |
|
||||||
txFuelRecipient.visible = true |
|
||||||
txDataLabel.text = "Contract arguments" |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
Label { |
|
||||||
id: txDataLabel |
|
||||||
text: "Contract code" |
|
||||||
} |
|
||||||
|
|
||||||
TextArea { |
|
||||||
id: codeView |
|
||||||
height: 300 |
|
||||||
anchors.topMargin: 5 |
|
||||||
Layout.fillWidth: true |
|
||||||
width: parent.width /2 |
|
||||||
onTextChanged: { |
|
||||||
contractFormReady() |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
TextField { |
|
||||||
id: txFuelRecipient |
|
||||||
placeholderText: "Contract address" |
|
||||||
validator: RegExpValidator { regExp: /[a-f0-9]{40}/ } |
|
||||||
visible: false |
|
||||||
width: 530 |
|
||||||
} |
|
||||||
|
|
||||||
Button { |
|
||||||
id: txButton |
|
||||||
/* enabled: false */ |
|
||||||
states: [ |
|
||||||
State { |
|
||||||
name: "READY" |
|
||||||
PropertyChanges { target: txButton; /*enabled: true*/} |
|
||||||
}, |
|
||||||
State { |
|
||||||
name: "NOTREADY" |
|
||||||
PropertyChanges { target: txButton; /*enabled:false*/} |
|
||||||
} |
|
||||||
] |
|
||||||
text: "Send" |
|
||||||
onClicked: { |
|
||||||
//this.enabled = false |
|
||||||
var res = eth.create(txFuelRecipient.text, txValue.text, txGas.text, txGasPrice.text, codeView.text) |
|
||||||
if(res[1]) { |
|
||||||
txResult.text = "Your contract <b>could not</b> be send over the network:\n<b>" |
|
||||||
txResult.text += res[1].error() |
|
||||||
txResult.text += "</b>" |
|
||||||
mainContractColumn.state = "ERROR" |
|
||||||
} else { |
|
||||||
txResult.text = "Your transaction has been submitted:\n" |
|
||||||
txOutput.text = res[0].address |
|
||||||
mainContractColumn.state = "DONE" |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
Text { |
|
||||||
id: txResult |
|
||||||
visible: false |
|
||||||
} |
|
||||||
TextField { |
|
||||||
id: txOutput |
|
||||||
visible: false |
|
||||||
width: 530 |
|
||||||
} |
|
||||||
Button { |
|
||||||
id: newTxButton |
|
||||||
visible: false |
|
||||||
text: "Create an other contract" |
|
||||||
onClicked: { |
|
||||||
this.visible = false |
|
||||||
txResult.text = "" |
|
||||||
txOutput.text = "" |
|
||||||
mainContractColumn.state = "SETUP" |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
Button { |
|
||||||
id: debugButton |
|
||||||
text: "Debug" |
|
||||||
onClicked: { |
|
||||||
var res = ui.debugTx("", txValue.text, txGas.text, txGasPrice.text, codeView.text) |
|
||||||
debugWindow.visible = true |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
@ -1,112 +0,0 @@ |
|||||||
import QtQuick 2.0 |
|
||||||
import QtQuick.Controls 1.0; |
|
||||||
import QtQuick.Layouts 1.0; |
|
||||||
import QtQuick.Dialogs 1.0; |
|
||||||
import QtQuick.Window 2.1; |
|
||||||
import QtQuick.Controls.Styles 1.1 |
|
||||||
import Ethereum 1.0 |
|
||||||
|
|
||||||
Component { |
|
||||||
id: newTransaction |
|
||||||
Column { |
|
||||||
id: simpleSendColumn |
|
||||||
states: [ |
|
||||||
State{ |
|
||||||
name: "ERROR" |
|
||||||
}, |
|
||||||
State { |
|
||||||
name: "DONE" |
|
||||||
PropertyChanges { target: txSimpleValue; visible:false} |
|
||||||
PropertyChanges { target: txSimpleRecipient; visible:false} |
|
||||||
PropertyChanges { target:newSimpleTxButton; visible:false} |
|
||||||
|
|
||||||
PropertyChanges { target: txSimpleResult; visible:true} |
|
||||||
PropertyChanges { target: txSimpleOutput; visible:true} |
|
||||||
PropertyChanges { target:newSimpleTxButton; visible:true} |
|
||||||
}, |
|
||||||
State { |
|
||||||
name: "SETUP" |
|
||||||
PropertyChanges { target: txSimpleValue; visible:true; text: ""} |
|
||||||
PropertyChanges { target: txSimpleRecipient; visible:true; text: ""} |
|
||||||
PropertyChanges { target: txSimpleButton; visible:true} |
|
||||||
PropertyChanges { target:newSimpleTxButton; visible:false} |
|
||||||
} |
|
||||||
] |
|
||||||
spacing: 5 |
|
||||||
anchors.leftMargin: 5 |
|
||||||
anchors.topMargin: 5 |
|
||||||
anchors.top: parent.top |
|
||||||
anchors.left: parent.left |
|
||||||
|
|
||||||
function checkFormState(){ |
|
||||||
if(txSimpleRecipient.text.length == 40 && txSimpleValue.text.length > 0) { |
|
||||||
txSimpleButton.state = "READY" |
|
||||||
}else{ |
|
||||||
txSimpleButton.state = "NOTREADY" |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
TextField { |
|
||||||
id: txSimpleRecipient |
|
||||||
placeholderText: "Recipient address" |
|
||||||
Layout.fillWidth: true |
|
||||||
validator: RegExpValidator { regExp: /[a-f0-9]{40}/ } |
|
||||||
width: 530 |
|
||||||
onTextChanged: { checkFormState() } |
|
||||||
} |
|
||||||
TextField { |
|
||||||
id: txSimpleValue |
|
||||||
width: 200 |
|
||||||
placeholderText: "Amount" |
|
||||||
anchors.rightMargin: 5 |
|
||||||
validator: RegExpValidator { regExp: /\d*/ } |
|
||||||
onTextChanged: { checkFormState() } |
|
||||||
} |
|
||||||
Button { |
|
||||||
id: txSimpleButton |
|
||||||
/*enabled: false*/ |
|
||||||
states: [ |
|
||||||
State { |
|
||||||
name: "READY" |
|
||||||
PropertyChanges { target: txSimpleButton; /*enabled: true*/} |
|
||||||
}, |
|
||||||
State { |
|
||||||
name: "NOTREADY" |
|
||||||
PropertyChanges { target: txSimpleButton; /*enabled: false*/} |
|
||||||
} |
|
||||||
] |
|
||||||
text: "Send" |
|
||||||
onClicked: { |
|
||||||
//this.enabled = false |
|
||||||
var res = eth.transact(txSimpleRecipient.text, txSimpleValue.text,"","","") |
|
||||||
if(res[1]) { |
|
||||||
txSimpleResult.text = "There has been an error broadcasting your transaction:" + res[1].error() |
|
||||||
} else { |
|
||||||
txSimpleResult.text = "Your transaction has been broadcasted over the network.\nYour transaction id is:" |
|
||||||
txSimpleOutput.text = res[0].hash |
|
||||||
this.visible = false |
|
||||||
simpleSendColumn.state = "DONE" |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
Text { |
|
||||||
id: txSimpleResult |
|
||||||
visible: false |
|
||||||
|
|
||||||
} |
|
||||||
TextField { |
|
||||||
id: txSimpleOutput |
|
||||||
visible: false |
|
||||||
width: 530 |
|
||||||
} |
|
||||||
Button { |
|
||||||
id: newSimpleTxButton |
|
||||||
visible: false |
|
||||||
text: "Create an other transaction" |
|
||||||
onClicked: { |
|
||||||
this.visible = false |
|
||||||
simpleSendColumn.state = "SETUP" |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
@ -0,0 +1,225 @@ |
|||||||
|
package ethui |
||||||
|
|
||||||
|
import ( |
||||||
|
"fmt" |
||||||
|
"github.com/ethereum/eth-go/ethchain" |
||||||
|
"github.com/ethereum/eth-go/ethutil" |
||||||
|
"github.com/go-qml/qml" |
||||||
|
"math/big" |
||||||
|
"strings" |
||||||
|
) |
||||||
|
|
||||||
|
type DebuggerWindow struct { |
||||||
|
win *qml.Window |
||||||
|
engine *qml.Engine |
||||||
|
lib *UiLib |
||||||
|
Db *Debugger |
||||||
|
} |
||||||
|
|
||||||
|
func NewDebuggerWindow(lib *UiLib) *DebuggerWindow { |
||||||
|
engine := qml.NewEngine() |
||||||
|
component, err := engine.LoadFile(lib.AssetPath("debugger/debugger.qml")) |
||||||
|
if err != nil { |
||||||
|
fmt.Println(err) |
||||||
|
|
||||||
|
return nil |
||||||
|
} |
||||||
|
|
||||||
|
win := component.CreateWindow(nil) |
||||||
|
db := &Debugger{win, make(chan bool), make(chan bool), true, false} |
||||||
|
|
||||||
|
return &DebuggerWindow{engine: engine, win: win, lib: lib, Db: db} |
||||||
|
} |
||||||
|
|
||||||
|
func (self *DebuggerWindow) Show() { |
||||||
|
context := self.engine.Context() |
||||||
|
context.SetVar("dbg", self) |
||||||
|
|
||||||
|
go func() { |
||||||
|
self.win.Show() |
||||||
|
self.win.Wait() |
||||||
|
}() |
||||||
|
} |
||||||
|
|
||||||
|
func (self *DebuggerWindow) SetCode(code string) { |
||||||
|
self.win.Set("codeText", code) |
||||||
|
} |
||||||
|
|
||||||
|
func (self *DebuggerWindow) SetData(data string) { |
||||||
|
self.win.Set("dataText", data) |
||||||
|
} |
||||||
|
func (self *DebuggerWindow) SetAsm(data string) { |
||||||
|
dis := ethchain.Disassemble(ethutil.FromHex(data)) |
||||||
|
for _, str := range dis { |
||||||
|
self.win.Root().Call("setAsm", str) |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
func (self *DebuggerWindow) Debug(valueStr, gasStr, gasPriceStr, scriptStr, dataStr string) { |
||||||
|
if !self.Db.done { |
||||||
|
self.Db.Q <- true |
||||||
|
} |
||||||
|
|
||||||
|
defer func() { |
||||||
|
if r := recover(); r != nil { |
||||||
|
self.Logf("compile FAULT: %v", r) |
||||||
|
} |
||||||
|
}() |
||||||
|
|
||||||
|
data := ethutil.StringToByteFunc(dataStr, func(s string) (ret []byte) { |
||||||
|
slice := strings.Split(dataStr, "\n") |
||||||
|
for _, dataItem := range slice { |
||||||
|
d := ethutil.FormatData(dataItem) |
||||||
|
ret = append(ret, d...) |
||||||
|
} |
||||||
|
return |
||||||
|
}) |
||||||
|
|
||||||
|
var err error |
||||||
|
script := ethutil.StringToByteFunc(scriptStr, func(s string) (ret []byte) { |
||||||
|
ret, err = ethutil.Compile(s) |
||||||
|
fmt.Printf("%x\n", ret) |
||||||
|
return |
||||||
|
}) |
||||||
|
|
||||||
|
if err != nil { |
||||||
|
self.Logln(err) |
||||||
|
|
||||||
|
return |
||||||
|
} |
||||||
|
|
||||||
|
dis := ethchain.Disassemble(script) |
||||||
|
self.win.Root().Call("clearAsm") |
||||||
|
self.win.Root().Call("clearLog") |
||||||
|
|
||||||
|
for _, str := range dis { |
||||||
|
self.win.Root().Call("setAsm", str) |
||||||
|
} |
||||||
|
|
||||||
|
gas := ethutil.Big(gasStr) |
||||||
|
gasPrice := ethutil.Big(gasPriceStr) |
||||||
|
// Contract addr as test address
|
||||||
|
keyPair := ethutil.GetKeyRing().Get(0) |
||||||
|
callerTx := ethchain.NewContractCreationTx(ethutil.Big(valueStr), gas, gasPrice, script) |
||||||
|
callerTx.Sign(keyPair.PrivateKey) |
||||||
|
|
||||||
|
state := self.lib.eth.BlockChain().CurrentBlock.State() |
||||||
|
account := self.lib.eth.StateManager().TransState().GetAccount(keyPair.Address()) |
||||||
|
contract := ethchain.MakeContract(callerTx, state) |
||||||
|
callerClosure := ethchain.NewClosure(account, contract, script, state, gas, gasPrice) |
||||||
|
|
||||||
|
block := self.lib.eth.BlockChain().CurrentBlock |
||||||
|
vm := ethchain.NewVm(state, self.lib.eth.StateManager(), ethchain.RuntimeVars{ |
||||||
|
Origin: account.Address(), |
||||||
|
BlockNumber: block.BlockInfo().Number, |
||||||
|
PrevHash: block.PrevHash, |
||||||
|
Coinbase: block.Coinbase, |
||||||
|
Time: block.Time, |
||||||
|
Diff: block.Difficulty, |
||||||
|
Value: ethutil.Big(valueStr), |
||||||
|
}) |
||||||
|
|
||||||
|
self.Db.done = false |
||||||
|
self.Logf("callsize %d", len(script)) |
||||||
|
go func() { |
||||||
|
ret, g, err := callerClosure.Call(vm, data, self.Db.halting) |
||||||
|
tot := new(big.Int).Mul(g, gasPrice) |
||||||
|
self.Logf("gas usage %v total price = %v (%v)", g, tot, ethutil.CurrencyToString(tot)) |
||||||
|
if err != nil { |
||||||
|
self.Logln("exited with errors:", err) |
||||||
|
} else { |
||||||
|
if len(ret) > 0 { |
||||||
|
self.Logf("exited: % x", ret) |
||||||
|
} else { |
||||||
|
self.Logf("exited: nil") |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
state.Reset() |
||||||
|
|
||||||
|
if !self.Db.interrupt { |
||||||
|
self.Db.done = true |
||||||
|
} else { |
||||||
|
self.Db.interrupt = false |
||||||
|
} |
||||||
|
}() |
||||||
|
} |
||||||
|
|
||||||
|
func (self *DebuggerWindow) Logf(format string, v ...interface{}) { |
||||||
|
self.win.Root().Call("setLog", fmt.Sprintf(format, v...)) |
||||||
|
} |
||||||
|
|
||||||
|
func (self *DebuggerWindow) Logln(v ...interface{}) { |
||||||
|
str := fmt.Sprintln(v...) |
||||||
|
self.Logf("%s", str[:len(str)-1]) |
||||||
|
} |
||||||
|
|
||||||
|
func (self *DebuggerWindow) Next() { |
||||||
|
self.Db.Next() |
||||||
|
} |
||||||
|
|
||||||
|
type Debugger struct { |
||||||
|
win *qml.Window |
||||||
|
N chan bool |
||||||
|
Q chan bool |
||||||
|
done, interrupt bool |
||||||
|
} |
||||||
|
|
||||||
|
type storeVal struct { |
||||||
|
Key, Value string |
||||||
|
} |
||||||
|
|
||||||
|
func (d *Debugger) halting(pc int, op ethchain.OpCode, mem *ethchain.Memory, stack *ethchain.Stack, stateObject *ethchain.StateObject) bool { |
||||||
|
d.win.Root().Call("setInstruction", pc) |
||||||
|
d.win.Root().Call("clearMem") |
||||||
|
d.win.Root().Call("clearStack") |
||||||
|
d.win.Root().Call("clearStorage") |
||||||
|
|
||||||
|
addr := 0 |
||||||
|
for i := 0; i+32 <= mem.Len(); i += 32 { |
||||||
|
d.win.Root().Call("setMem", memAddr{fmt.Sprintf("%03d", addr), fmt.Sprintf("% x", mem.Data()[i:i+32])}) |
||||||
|
addr++ |
||||||
|
} |
||||||
|
|
||||||
|
for _, val := range stack.Data() { |
||||||
|
d.win.Root().Call("setStack", val.String()) |
||||||
|
} |
||||||
|
|
||||||
|
stateObject.State().EachStorage(func(key string, node *ethutil.Value) { |
||||||
|
d.win.Root().Call("setStorage", storeVal{fmt.Sprintf("% x", key), fmt.Sprintf("% x", node.Str())}) |
||||||
|
}) |
||||||
|
|
||||||
|
out: |
||||||
|
for { |
||||||
|
select { |
||||||
|
case <-d.N: |
||||||
|
break out |
||||||
|
case <-d.Q: |
||||||
|
d.interrupt = true |
||||||
|
d.clearBuffers() |
||||||
|
|
||||||
|
return false |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
return true |
||||||
|
} |
||||||
|
|
||||||
|
func (d *Debugger) clearBuffers() { |
||||||
|
out: |
||||||
|
// drain
|
||||||
|
for { |
||||||
|
select { |
||||||
|
case <-d.N: |
||||||
|
case <-d.Q: |
||||||
|
default: |
||||||
|
break out |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
func (d *Debugger) Next() { |
||||||
|
if !d.done { |
||||||
|
d.N <- true |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue