github.com/gop9/olt@v0.0.0-20200202132135-d956aad50b08/gio/app/internal/log/log_android.go (about) 1 // SPDX-License-Identifier: Unlicense OR MIT 2 3 package log 4 5 /* 6 #cgo LDFLAGS: -llog 7 8 #include <stdlib.h> 9 #include <android/log.h> 10 */ 11 import "C" 12 13 import ( 14 "bufio" 15 "log" 16 "os" 17 "runtime" 18 "syscall" 19 "unsafe" 20 ) 21 22 func init() { 23 // Android's logcat already includes timestamps. 24 log.SetFlags(log.Flags() &^ log.LstdFlags) 25 logFd(os.Stdout.Fd()) 26 logFd(os.Stderr.Fd()) 27 } 28 29 func logFd(fd uintptr) { 30 r, w, err := os.Pipe() 31 if err != nil { 32 panic(err) 33 } 34 if err := syscall.Dup3(int(w.Fd()), int(fd), syscall.O_CLOEXEC); err != nil { 35 panic(err) 36 } 37 go func() { 38 tag := C.CString("gio") 39 defer C.free(unsafe.Pointer(tag)) 40 // 1024 is the truncation limit from android/log.h, plus a \n. 41 lineBuf := bufio.NewReaderSize(r, 1024) 42 // The buffer to pass to C, including the terminating '\0'. 43 buf := make([]byte, lineBuf.Size()+1) 44 cbuf := (*C.char)(unsafe.Pointer(&buf[0])) 45 for { 46 line, _, err := lineBuf.ReadLine() 47 if err != nil { 48 break 49 } 50 copy(buf, line) 51 buf[len(line)] = 0 52 C.__android_log_write(C.ANDROID_LOG_INFO, tag, cbuf) 53 } 54 // The garbage collector doesn't know that w's fd was dup'ed. 55 // Avoid finalizing w, and thereby avoid its finalizer closing its fd. 56 runtime.KeepAlive(w) 57 }() 58 }