📄 console.tcl
字号:
# {{{ Banner
# ============================================================================
#
# console.tcl
#
# Console output support for the eCos synthetic target I/O auxiliary
#
# ============================================================================
# ####COPYRIGHTBEGIN####
#
# ----------------------------------------------------------------------------
# Copyright (C) 2002 Bart Veer
#
# This file is part of the eCos host tools.
#
# This program is free software; you can redistribute it and/or modify it
# under the terms of the GNU General Public License as published by the Free
# Software Foundation; either version 2 of the License, or (at your option)
# any later version.
#
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
# more details.
#
# You should have received a copy of the GNU General Public License along with
# this program; if not, write to the Free Software Foundation, Inc.,
# 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#
# ----------------------------------------------------------------------------
#
# ####COPYRIGHTEND####
# ============================================================================
# #####DESCRIPTIONBEGIN####
#
# Author(s): bartv
# Contact(s): bartv
# Date: 2002/08/07
# Version: 0.01
# Description:
# Implementation of the console device. This script should only ever
# be run from inside the ecosynth auxiliary.
#
# ####DESCRIPTIONEND####
# ============================================================================
# }}}
# The console device is pretty simple. There can only ever be one
# instance of the console, and it does not take any initialization
# data from the target or from command-line arguments. It does
# look for entries in the target definition file, to set up
# colours for console output, and to install additional regexp-based
# filters. The only type of request that can go to the console device
# is to write some text.
namespace eval ::console {
variable _pending_output ""
variable filter_count 0
array set filters [list]
proc instantiate { id name data } {
# There is only console so no name is expected, and the target
# cannot provide any initialization data.
if { ("" != $name) || ("" != $data) } {
synth::report_error "The target has passed invalid data when instantiating the console device.\n"
return ""
}
# There are no command line arguments to be processed and consumed.
# Look for and consume target definition entries related to the console.
# These are only actually applicable when running in GUI mode, but
# should always be consumed.
set console_appearance ""
if { [synth::tdf_has_device "console"] } {
if { [synth::tdf_has_option "console" "appearance"] } {
set console_appearance [synth::tdf_get_option "console" "appearance"]
}
if { [synth::tdf_has_option "console" "filter"] } {
set tdf_filters [synth::tdf_get_options "console" "filter"]
foreach filter $tdf_filters {
if { 2 > [llength $filter] } {
set msg "Invalid entry in target definition file $synth::target_definition\n"
append msg " Device console, option filter takes at least two arguments, a name and a regular expression.\n"
synth::report_error $msg
} else {
# Attempt some minimal validation of the regexp
set name [lindex $filter 0]
set regexp [lindex $filter 1]
set error ""
if { [catch { regexp -- $regexp "Hello world\n" } error] } {
set msg "Invalid entry in target definition file $synth::target_definition\n"
append msg " Device console, filter $name, invalid regular expression\n $error\n"
synth::report_error $msg
} else {
set ::console::filters($::console::filter_count,name) $name
set ::console::filters($::console::filter_count,regexp) $regexp
set ::console::filters($::console::filter_count,appearance) [lrange $filter 2 end]
incr ::console::filter_count
}
}
}
}
}
# If the GUI is enabled then set up a filter for the console, and
# any additional filters specified in the target definition file
# for e.g. trace output.
if { $synth::flag_gui } {
if { [synth::filter_exists "console" ] } {
synth::report_warning "The console device script [info script] cannot create a filter for \"console\".\nThis filter already exists.\n"
} elseif { "" == $console_appearance } {
synth::filter_add "console"
} else {
array set parsed_options [list]
set message ""
if { ![synth::filter_parse_options $console_appearance parsed_options message] } {
synth::report_error \
"Invalid entry in target definition file $synth::target_definition\
\n synth_device \"console\", entry \"appearance\"\n$message"
} else {
synth::filter_add_parsed "console" parsed_options
}
}
for { set i 0 } { $i < $::console::filter_count } { incr i } {
set name $::console::filters($i,name)
set appearance $::console::filters($i,appearance)
array unset parsed_options
array set parsed_options [list]
if { [synth::filter_exists $name] } {
synth::report_warning "The console device script [info script] cannot create a filter for \"$name\".\nThis filter already exists.\n"
} else {
set message ""
if { ![synth::filter_parse_options $appearance parsed_options message] } {
synth::report_error \
"Invalid entry in target definition file $synth::target_definition\
\n synth_device \"console\", entry filter $name\n$message"
} else {
synth::filter_add_parsed $name parsed_options
}
}
}
}
# An instantiation function should return a handler for further requests
# to this device instance.
return console::handle_request
}
proc handle_request { id reqcode arg1 arg2 reqdata reqlen reply_len } {
# Unfortunately the main eCos diagnostic code assumes it is
# talking to a tty in raw mode, since typically the output
# will go via the gdb output window. Hence it will have inserted
# carriage returns which are best filtered out here.
set reqdata [string map {"\r" " "} $reqdata]
# The output should be processed one line at a time, to make it
# easier to write the regexp filters.
append console::_pending_output $reqdata
while { -1 != [string first "\n" $console::_pending_output] } {
set regexp_matched 0
set index [string first "\n" $console::_pending_output]
set line [string range $console::_pending_output 0 $index]
set ::console::_pending_output [string range $console::_pending_output [expr $index + 1] end]
for { set i 0 } { !$regexp_matched && ($i < $console::filter_count) } { incr i } {
if { [regexp -- $console::filters($i,regexp) $line] } {
synth::output $line $console::filters($i,name)
set regexp_matched 1
}
}
if { ! $regexp_matched } {
synth::output $line "console"
}
}
}
# Deal with the case where eCos has exited after sending only part
# of a line, which is still pending. In practice this has no
# effect at present because the data is still buffered inside
# eCos.
proc _flush { arg_list } {
if { "" != $console::_pending_output } {
synth::output "$console::_pending_output\n" "console"
}
}
synth::hook_add "ecos_exit" console::_flush
}
return console::instantiate
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -