英文:
How do you get the terminal size in Go?
问题
在Go语言中,如何获取终端大小?在C语言中,可以这样写:
struct ttysize ts;
ioctl(0, TIOCGWINSZ, &ts);
但是在Go语言中,如何访问TIOCGWINSZ呢?
英文:
How do I get the terminal size in Go. In C it would look like this:
struct ttysize ts;
ioctl(0, TIOCGWINSZ, &ts);
But how to i access TIOCGWINSZ in Go
答案1
得分: 3
cgo编译器目前无法处理C函数中的可变参数和C头文件中的宏,因此您无法简单地执行以下操作:
// #include <sys/ioctl.h>
// typedef struct ttysize ttysize;
import "C"
func GetWinSz() {
var ts C.ttysize;
C.ioctl(0,C.TIOCGWINSZ,&ts)
}
为了解决宏的问题,您可以使用一个常量,例如:
// #include <sys/ioctl.h>
// typedef struct ttysize ttysize;
import "C"
const TIOCGWINSZ C.ulong = 0x5413; // 来自Jed Smith的答案中的值
func GetWinSz() {
var ts C.ttysize;
C.ioctl(0,TIOCGWINSZ,&ts)
}
然而,cgo仍然无法处理ioctl原型中的...。您最好的选择是使用一个接受特定数量参数的C函数来包装ioctl,并将其链接进来。作为一种hack,您可以在import "C"之前的注释中执行此操作:
// #include <sys/ioctl.h>
// typedef struct ttysize ttysize;
// void myioctl(int i, unsigned long l, ttysize * t){ioctl(i,l,t);}
import "C"
const TIOCGWINSZ C.ulong = 0x5413; // 来自Jed Smith的答案中的值
func GetWinSz() {
var ts C.ttysize;
C.myioctl(0,TIOCGWINSZ,&ts)
}
我没有测试过这个,但类似的方法应该可以工作。
英文:
The cgo compiler can't handle variable arguments in a c function and macros in c header files at present, so you can't do a simple
// #include <sys/ioctl.h>
// typedef struct ttysize ttysize;
import "C"
func GetWinSz() {
var ts C.ttysize;
C.ioctl(0,C.TIOCGWINSZ,&ts)
}
To get around the macros use a constant, so
// #include <sys/ioctl.h>
// typedef struct ttysize ttysize;
import "C"
const TIOCGWINSZ C.ulong = 0x5413; // Value from Jed Smith's answer
func GetWinSz() {
var ts C.ttysize;
C.ioctl(0,TIOCGWINSZ,&ts)
}
However cgo will still barf on the ... in ioctl's prototype. Your best bet would be to wrap ioctl with a c function taking a specific number of arguments and link that in. As a hack you can do that in the comment above import "C"
// #include <sys/ioctl.h>
// typedef struct ttysize ttysize;
// void myioctl(int i, unsigned long l, ttysize * t){ioctl(i,l,t);}
import "C"
const TIOCGWINSZ C.ulong = 0x5413; // Value from Jed Smith's answer
func GetWinSz() {
var ts C.ttysize;
C.myioctl(0,TIOCGWINSZ,&ts)
}
I've not tested this, but something similar should work.
答案2
得分: 1
最好的方法是使用syscall包。syscall包没有定义ioctl函数,因为它可以执行许多不同的操作,但你仍然可以像这样调用它:
syscall.Syscall(syscall.SYS_IOCTL, uintptr(fd), uintptr(TIOCGWINSZ), uintptr(unsafe.Pointer(&ts)))
剩下的两件事是复制winsize结构和所需的常量。用于此的工具是godefs,它将根据C头文件中的结构和常量生成一个.go源文件。创建一个termios.c文件,内容如下:
#include <termios.h>
enum {
$TIOCGWINSZ = TIOCGWINSZ
};
typedef winsize $winsize;
然后运行
godefs -gpackagename termios.c > termios.go
现在你应该拥有获取终端大小所需的一切。设置大小只需在termios.c中添加另一个常量即可。
英文:
The best way to do this would be to use the syscall package. The syscall package does not define an ioctl function because it just does so many different things, but you can still call it like this:
syscall.Syscall(syscall.SYS_IOCTL, uintptr(fd), uintptr(TIOCGWINSZ), uintptr(unsafe.Pointer(&ts)))
The two things left are to duplicate the winsize structure and the constant you need. The tool for this is godefs, which will generate a .go source file by looking at structures and constants in C headers. Create a termios.c file that looks like this:
#include <termios.h>
enum {
$TIOCGWINSZ = TIOCGWINSZ
};
typedef winsize $winsize;
Now run
godefs -gpackagename termios.c > termios.go
Now you should have everything you need to get the terminal size. Setting the size is as simple as adding another constant to termios.c.
答案3
得分: 1
read: http://www.darkcoding.net/software/pretty-command-line-console-output-on-unix-in-python-and-go-lang/
const (
TIOCGWINSZ = 0x5413
TIOCGWINSZ_OSX = 1074295912
)
type window struct {
Row uint16
Col uint16
Xpixel uint16
Ypixel uint16
}
func terminalWidth() (int, error) {
w := new(window)
tio := syscall.TIOCGWINSZ
if runtime.GOOS == "darwin" {
tio = TIOCGWINSZ_OSX
}
res, _, err := syscall.Syscall(syscall.SYS_IOCTL,
uintptr(syscall.Stdin),
uintptr(tio),
uintptr(unsafe.Pointer(w)),
)
if int(res) == -1 {
return 0, err
}
return int(w.Col), nil
}
英文:
read: http://www.darkcoding.net/software/pretty-command-line-console-output-on-unix-in-python-and-go-lang/
const (
TIOCGWINSZ = 0x5413
TIOCGWINSZ_OSX = 1074295912
)
type window struct {
Row uint16
Col uint16
Xpixel uint16
Ypixel uint16
}
func terminalWidth() (int, error) {
w := new(window)
tio := syscall.TIOCGWINSZ
if runtime.GOOS == "darwin" {
tio = TIOCGWINSZ_OSX
}
res, _, err := syscall.Syscall(syscall.SYS_IOCTL,
uintptr(syscall.Stdin),
uintptr(tio),
uintptr(unsafe.Pointer(w)),
)
if int(res) == -1 {
return 0, err
}
return int(w.Col), nil
}
答案4
得分: 0
从文档上看,似乎还没有做太多的工作 - 实际上,我根本找不到ioctl
。
对于这样一个刚刚起步的语言来说,可以说你正在踏上未曾涉足的领域。TIOCGWINSZ
本身只是一个常量整数(我在Linux源代码中找到了它的值):
#define TIOCGWINSZ 0x5413
祝你好运。
英文:
It doesn't look like much work has been done on this yet from a casual glance of the documentation -- in fact, I can't find ioctl
at all.
With a language this early in its infancy, it's safe to say that you're treading unwalked ground. TIOCGWINSZ
, itself, is just a constant integer (I found its value in the Linux source code):
#define TIOCGWINSZ 0x5413
Good luck, though.
通过集体智慧和协作来改善编程学习和解决问题的方式。致力于成为全球开发者共同参与的知识库,让每个人都能够通过互相帮助和分享经验来进步。
评论