aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorColin Cross <ccross@android.com>2019-06-11 23:01:36 -0700
committerLuca Stefani <luca.stefani.ge1@gmail.com>2019-09-04 15:14:54 +0200
commit71fb7f3ee21a380f1187de83ae080f95a93f1fb6 (patch)
tree52c17f0425254685453b0235ecf79562f086c635
parentb63b662db3771e5991ebaa75b2ad81b3846f83ff (diff)
downloadbuild_soong-71fb7f3ee21a380f1187de83ae080f95a93f1fb6.tar.gz
build_soong-71fb7f3ee21a380f1187de83ae080f95a93f1fb6.tar.bz2
build_soong-71fb7f3ee21a380f1187de83ae080f95a93f1fb6.zip
Use SIGWINCH to update terminal size
Instead of reading the terminal size on every status update, register for SIGWINCH to read and store the size when it changes. This reapplies I555ad21a31a2c924ab0ca681e0c8f00df42a370a with a fix for a race condition in TestSmartStatusOutputWidthChange. Test: status_test.go Change-Id: If342cb4cc8e4ed57af9e3bb417758348c9c41247
-rw-r--r--ui/terminal/smart_status.go66
-rw-r--r--ui/terminal/status_test.go6
2 files changed, 61 insertions, 11 deletions
diff --git a/ui/terminal/smart_status.go b/ui/terminal/smart_status.go
index 8fa9effc..82c04d46 100644
--- a/ui/terminal/smart_status.go
+++ b/ui/terminal/smart_status.go
@@ -17,8 +17,11 @@ package terminal
import (
"fmt"
"io"
+ "os"
+ "os/signal"
"strings"
"sync"
+ "syscall"
"android/soong/ui/status"
)
@@ -30,18 +33,30 @@ type smartStatusOutput struct {
lock sync.Mutex
haveBlankLine bool
+
+ termWidth int
+ sigwinch chan os.Signal
+ sigwinchHandled chan bool
}
// NewSmartStatusOutput returns a StatusOutput that represents the
// current build status similarly to Ninja's built-in terminal
// output.
func NewSmartStatusOutput(w io.Writer, formatter formatter) status.StatusOutput {
- return &smartStatusOutput{
+ s := &smartStatusOutput{
writer: w,
formatter: formatter,
haveBlankLine: true,
+
+ sigwinch: make(chan os.Signal),
}
+
+ s.updateTermSize()
+
+ s.startSigwinch()
+
+ return s
}
func (s *smartStatusOutput) Message(level status.MsgLevel, message string) {
@@ -101,6 +116,8 @@ func (s *smartStatusOutput) Flush() {
s.lock.Lock()
defer s.lock.Unlock()
+ s.stopSigwinch()
+
s.requestLine()
}
@@ -137,16 +154,8 @@ func (s *smartStatusOutput) statusLine(str string) {
// Limit line width to the terminal width, otherwise we'll wrap onto
// another line and we won't delete the previous line.
- //
- // Run this on every line in case the window has been resized while
- // we're printing. This could be optimized to only re-run when we get
- // SIGWINCH if it ever becomes too time consuming.
- if max, ok := termWidth(s.writer); ok {
- if len(str) > max {
- // TODO: Just do a max. Ninja elides the middle, but that's
- // more complicated and these lines aren't that important.
- str = str[:max]
- }
+ if s.termWidth > 0 {
+ str = s.elide(str)
}
// Move to the beginning on the line, turn on bold, print the output,
@@ -156,3 +165,38 @@ func (s *smartStatusOutput) statusLine(str string) {
fmt.Fprint(s.writer, start, str, end)
s.haveBlankLine = false
}
+
+func (s *smartStatusOutput) elide(str string) string {
+ if len(str) > s.termWidth {
+ // TODO: Just do a max. Ninja elides the middle, but that's
+ // more complicated and these lines aren't that important.
+ str = str[:s.termWidth]
+ }
+
+ return str
+}
+
+func (s *smartStatusOutput) startSigwinch() {
+ signal.Notify(s.sigwinch, syscall.SIGWINCH)
+ go func() {
+ for _ = range s.sigwinch {
+ s.lock.Lock()
+ s.updateTermSize()
+ s.lock.Unlock()
+ if s.sigwinchHandled != nil {
+ s.sigwinchHandled <- true
+ }
+ }
+ }()
+}
+
+func (s *smartStatusOutput) stopSigwinch() {
+ signal.Stop(s.sigwinch)
+ close(s.sigwinch)
+}
+
+func (s *smartStatusOutput) updateTermSize() {
+ if w, ok := termWidth(s.writer); ok {
+ s.termWidth = w
+ }
+}
diff --git a/ui/terminal/status_test.go b/ui/terminal/status_test.go
index a87a7f07..106d6515 100644
--- a/ui/terminal/status_test.go
+++ b/ui/terminal/status_test.go
@@ -17,6 +17,7 @@ package terminal
import (
"bytes"
"fmt"
+ "syscall"
"testing"
"android/soong/ui/status"
@@ -252,6 +253,8 @@ func actionWithOuptutWithAnsiCodes(stat status.StatusOutput) {
func TestSmartStatusOutputWidthChange(t *testing.T) {
smart := &fakeSmartTerminal{termWidth: 40}
stat := NewStatusOutput(smart, "", false)
+ smartStat := stat.(*smartStatusOutput)
+ smartStat.sigwinchHandled = make(chan bool)
runner := newRunner(stat, 2)
@@ -260,6 +263,9 @@ func TestSmartStatusOutputWidthChange(t *testing.T) {
runner.startAction(action)
smart.termWidth = 30
+ // Fake a SIGWINCH
+ smartStat.sigwinch <- syscall.SIGWINCH
+ <-smartStat.sigwinchHandled
runner.finishAction(result)
stat.Flush()