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