我想做什么?
我正在尝试构建一个 Python Qt Gui,它可以让我轻松地为微控制器运行/调试嵌入式程序。作为编译器,我使用的是 sdcc。Sdcc 还包含一个模拟器。
在代码中,我启动了一个子 QProcess,它将运行微控制器仿真器 ( s51
)。
在模拟器中,模拟从输入 r + RETURN ( "r\n"
) 开始。
按 RETURN ( ) 可以随时停止模拟"\n"
。
我的问题
我的问题是我的代码无法到达第二个 RETURN。换句话说,模拟无限期地继续。一种或另一种方式,子进程永远不会收到第二个 RETURN。
重现步骤
- 安装 Python、PyQt 和 sdcc。
- 将下面的代码复制/粘贴到一个 emtpy 文件中并运行它。
- 最终更改 s51 可执行文件的路径。
- 单击“开始”按钮。
- 在下面的 Input 中输入
r
,然后按 Return 键。现在应该开始模拟。 - 当您再次按下 Return 键时,模拟应该会停止。但事实并非如此。
笔记:
我已经在 Fedora Linux 上编写了代码。如果您正在运行 Ubuntu,则可能必须将可执行文件的路径更改为
/usr/bin/s51
(我不确定确切的路径)。该代码在其他一些控制台应用程序(如 bash、python、ifconfig、echo、gcc、...
我相信这个问题与 sdcc 在阅读“r\n”命令(使用 strace 找到)后立即执行 ioct 有关:
ioctl(0, SNDCTL_TMR_TIMEBASE or SNDRV_TIMER_IOCTL_NEXT_DEVICE or TCGETS, {B38400 opost isig icanon echo ...}) = 0
EDIT1:我正在运行 Qt 4.8.3
EDIT2:简化并修复了代码+添加了屏幕截图
截屏
模拟应该在第二次 RETURN 后停止。
(我输入了h run\n
, run\n
, \n
, \n
, \help
)
截图可以在这里找到(声望不够)
简单控制台.py:
#!/usr/bin/env python3
import sys
from PyQt4 import Qt, QtGui, QtCore
class SimpleConsole(QtGui.QMainWindow):
def __init__(self):
QtGui.QMainWindow.__init__(self)
self.setup()
self.process = None
def setup(self):
l = QtGui.QHBoxLayout()
lbl = QtGui.QLabel('Command:')
self.processExecutable = QtGui.QLineEdit()
self.processExecutable.setEnabled(True)
self.processExecutable.returnPressed.connect(self.startStopProcess)
self.processExecutable.setText('/usr/libexec/sdcc/s51')
lbl.setBuddy(self.processExecutable)
l.addWidget(lbl)
l.addWidget(self.processExecutable)
self.processOutputWidget = QtGui.QTextEdit()
self.processOutputWidget.setReadOnly(True)
self.processInputWidget = QtGui.QLineEdit()
self.processInputWidget.setEnabled(False)
self.processInputWidget.returnPressed.connect(self.inputForProcess)
main = QtGui.QVBoxLayout()
main.addLayout(l)
main.addWidget(self.processOutputWidget)
main.addWidget(self.processInputWidget)
widget = QtGui.QWidget()
widget.setLayout(main)
self.setCentralWidget(widget)
self.setStatusBar(QtGui.QStatusBar())
self.statusBarLabel = QtGui.QLabel()
self.statusBar().addWidget(self.statusBarLabel)
self.toolbar = QtGui.QToolBar()
self.addToolBar(self.toolbar)
self.startStopAction = self.toolbar.addAction('Start')
self.startStopAction.triggered[bool].connect(self.startStopProcess)
def closeEvent(self, event):
self.stopProcess()
@QtCore.pyqtSlot(QtCore.QProcess.ProcessState)
def processStateChanged(self, newState):
self.processInputWidget.setEnabled(newState == QtCore.QProcess.Running)
pid = self.process.pid()
if newState == QtCore.QProcess.Running:
self.startStopAction.setText('Stop')
self.startStopAction.setEnabled(True)
self.processExecutable.setEnabled(False)
elif newState == QtCore.QProcess.NotRunning:
self.processInputWidget.setEnabled(False)
self.processInputWidget.clear()
self.startStopAction.setText('Start')
self.startStopAction.setEnabled(True)
self.processExecutable.setEnabled(True)
self.process = None
status = {QtCore.QProcess.NotRunning:'not running',
QtCore.QProcess.Starting:'starting',
QtCore.QProcess.Running:'running'}
self.statusBarLabel.setText('Process state change: {newState}. (pid={pid})'.format(newState=status[newState], pid=pid))
@QtCore.pyqtSlot()
def startStopProcess(self):
if self.process:
self.stopProcess()
else:
self.processOutputWidget.clear()
self.process = QtCore.QProcess(self)
self.process.stateChanged.connect(self.processStateChanged)
self.process.readyReadStandardOutput.connect(self.processOutputRead)
self.process.readyReadStandardError.connect(self.processErrorRead)
exe = self.processExecutable.text()
self.process.start(exe, QtCore.QIODevice.ReadWrite)
def moveCursorToEnd(self):
self.processOutputWidget.moveCursor(QtGui.QTextCursor.End, QtGui.QTextCursor.MoveAnchor)
@QtCore.pyqtSlot()
def inputForProcess(self):
cmd = self.processInputWidget.text() + '\n'
self.moveCursorToEnd()
self.processOutputWidget.insertPlainText(cmd)
self.process.writeData(bytearray(cmd, encoding='utf-8'))
self.process.waitForBytesWritten()
self.processInputWidget.clear()
@QtCore.pyqtSlot()
def processOutputRead(self):
bytesOut = bytes(self.process.readAllStandardOutput()).decode('utf-8')
self.moveCursorToEnd()
self.processOutputWidget.insertPlainText(bytesOut)
@QtCore.pyqtSlot()
def processErrorRead(self):
bytesError = bytes(self.process.readAllStandardError()).decode('utf-8')
self.moveCursorToEnd()
self.processOutputWidget.insertPlainText(bytesError)
def stopProcess(self):
if self.process:
self.process.closeWriteChannel()
if not self.process.waitForFinished(500):
self.process.terminate()
if __name__ == '__main__':
app = QtGui.QApplication(sys.argv)
w = SimpleConsole()
w.show()
app.exec_()