1

我想做什么?

我正在尝试构建一个 Python Qt Gui,它可以让我轻松地为微控制器运行/调试嵌入式程序。作为编译器,我使用的是 sdcc。Sdcc 还包含一个模拟器。

在代码中,我启动了一个子 QProcess,它将运行微控制器仿真器 ( s51)。

在模拟器中,模拟从输入 r + RETURN ( "r\n") 开始。

按 RETURN ( ) 可以随时停止模拟"\n"

我的问题

我的问题是我的代码无法到达第二个 RETURN。换句话说,模拟无限期地继续。一种或另一种方式,子进程永远不会收到第二个 RETURN。

重现步骤

  1. 安装 Python、PyQt 和 sdcc。
  2. 将下面的代码复制/粘贴到一个 emtpy 文件中并运行它。
  3. 最终更改 s51 可执行文件的路径。
  4. 单击“开始”按钮。
  5. 在下面的 Input 中输入r,然后按 Return 键。现在应该开始模拟。
  6. 当您再次按下 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_()
4

0 回答 0