婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > Golang標準庫syscall詳解(什么是系統調用)

Golang標準庫syscall詳解(什么是系統調用)

熱門標簽:揚州電銷外呼系統軟件 百度地圖標注位置網站 武漢百應人工智能電銷機器人 電腦外呼系統輻射大嗎 上海企業外呼系統排名 智能語音電銷的機器人 開通400電話申請流程 如何利用高德地圖標注家 400手機電話免費辦理

一、什么是系統調用

In computing, a system call is the programmatic way in which a computer program requests a service from the kernel of the operating system it is executed on. This may include hardware-related services (for example, accessing a hard disk drive), creation and execution of new processes, and communication with integral kernel services such as process scheduling. System calls provide an essential interface between a process and the operating system.

系統調用是程序向操作系統內核請求服務的過程,通常包含硬件相關的服務(例如訪問硬盤),創建新進程等。系統調用提供了一個進程和操作系統之間的接口。

二、Golang標準庫-syscall

syscall包包含一個指向底層操作系統原語的接口。

注意:該軟件包已被鎖定。標準以外的代碼應該被遷移到golang.org/x/sys存儲庫中使用相應的軟件包。這也是應用新系統或版本所需更新的地方。 Signal , Errno 和 SysProcAttr 在 golang.org/x/sys 中尚不可用,并且仍然必須從 syscall 程序包中引用。有關更多信息,請參見 https://golang.org/s/go1.4-syscall。

https://pkg.go.dev/golang.org/x/sys
該存儲庫包含用于與操作系統進行低級交互的補充Go軟件包。

1. syscall無處不在

舉個最常用的例子, fmt.Println(“hello world”), 這里就用到了系統調用 write, 我們翻一下源碼。

func Println(a ...interface{}) (n int, err error) {
	return Fprintln(os.Stdout, a...)
}
Stdout = NewFile(uintptr(syscall.Stdout), "/dev/stdout")
 
func (f *File) write(b []byte) (n int, err error) {
    if len(b) == 0 {
        return 0, nil
    }
    // 實際的write方法,就是調用syscall.Write()
    return fixCount(syscall.Write(f.fd, b))
}

2. syscall demo舉例:

 go版本的strace Strace

strace 是用于查看進程系統調用的工具, 一般使用方法如下:

strace -c 用于統計各個系統調用的次數

[root@localhost ~]# strace -c echo hello
hello
% time     seconds  usecs/call     calls    errors syscall
------ ----------- ----------- --------- --------- ----------------
  0.00    0.000000           0         1           read
  0.00    0.000000           0         1           write
  0.00    0.000000           0         3           open
  0.00    0.000000           0         5           close
  0.00    0.000000           0         4           fstat
  0.00    0.000000           0         9           mmap
  0.00    0.000000           0         4           mprotect
  0.00    0.000000           0         2           munmap
  0.00    0.000000           0         4           brk
  0.00    0.000000           0         1         1 access
  0.00    0.000000           0         1           execve
  0.00    0.000000           0         1           arch_prctl
------ ----------- ----------- --------- --------- ----------------
100.00    0.000000                    36         1 total
[root@localhost ~]#

stace 的實現原理是系統調用 ptrace, 我們來看下 ptrace 是什么。

man page 描述如下:

The ptrace() system call provides a means by which one process (the “tracer”) may observe and control the execution of another process (the “tracee”), and examine and change the tracee's memory and registers. It is primarily used to implement breakpoint debuggingand system call tracing.

簡單來說有三大能力:

追蹤系統調用
讀寫內存和寄存器
向被追蹤程序傳遞信號

ptrace接口:

int ptrace(int request, pid_t pid, caddr_t addr, int data);
 
request包含:
PTRACE_ATTACH
PTRACE_SYSCALL
PTRACE_PEEKTEXT, PTRACE_PEEKDATA
等

tracer 使用 PTRACE_ATTACH 命令,指定需要追蹤的PID。緊接著調用 PTRACE_SYSCALL。
tracee 會一直運行,直到遇到系統調用,內核會停止執行。 此時,tracer 會收到 SIGTRAP 信號,tracer 就可以打印內存和寄存器中的信息了。

接著,tracer 繼續調用 PTRACE_SYSCALL, tracee 繼續執行,直到 tracee退出當前的系統調用。
需要注意的是,這里在進入syscall和退出syscall時,tracer都會察覺。

go版本的strace

了解以上內容后,presenter 現場實現了一個go版本的strace, 需要在 linux amd64 環境編譯。
https://github.com/silentred/gosys

// strace.go

package main
 
import (
    "fmt"
    "os"
    "os/exec"
    "syscall"
)
 
func main() {
    var err error
    var regs syscall.PtraceRegs
    var ss syscallCounter
    ss = ss.init()
 
    fmt.Println("Run: ", os.Args[1:])
 
    cmd := exec.Command(os.Args[1], os.Args[2:]...)
    cmd.Stderr = os.Stderr
    cmd.Stdout = os.Stdout
    cmd.Stdin = os.Stdin
    cmd.SysProcAttr = syscall.SysProcAttr{
        Ptrace: true,
    }
 
    cmd.Start()
    err = cmd.Wait()
    if err != nil {
        fmt.Printf("Wait err %v \n", err)
    }
 
    pid := cmd.Process.Pid
    exit := true
 
    for {
        // 記得 PTRACE_SYSCALL 會在進入和退出syscall時使 tracee 暫停,所以這里用一個變量控制,RAX的內容只打印一遍
        if exit {
            err = syscall.PtraceGetRegs(pid, regs)
            if err != nil {
                break
            }
            //fmt.Printf("%#v \n",regs)
            name := ss.getName(regs.Orig_rax)
            fmt.Printf("name: %s, id: %d \n", name, regs.Orig_rax)
            ss.inc(regs.Orig_rax)
        }
        // 上面Ptrace有提到的一個request命令
        err = syscall.PtraceSyscall(pid, 0)
        if err != nil {
            panic(err)
        }
        // 猜測是等待進程進入下一個stop,這里如果不等待,那么會打印大量重復的調用函數名
        _, err = syscall.Wait4(pid, nil, 0, nil)
        if err != nil {
            panic(err)
        }
 
        exit = !exit
    }
 
    ss.print()
}

// 用于統計信息的counter, syscallcounter.go

package main
 
import (
    "fmt"
    "os"
    "text/tabwriter"
 
    "github.com/seccomp/libseccomp-golang"
)
 
type syscallCounter []int
 
const maxSyscalls = 303
 
func (s syscallCounter) init() syscallCounter {
    s = make(syscallCounter, maxSyscalls)
    return s
}
 
func (s syscallCounter) inc(syscallID uint64) error {
    if syscallID > maxSyscalls {
        return fmt.Errorf("invalid syscall ID (%x)", syscallID)
    }
 
    s[syscallID]++
    return nil
}
 
func (s syscallCounter) print() {
    w := tabwriter.NewWriter(os.Stdout, 0, 0, 8, ' ', tabwriter.AlignRight|tabwriter.Debug)
    for k, v := range s {
        if v > 0 {
            name, _ := seccomp.ScmpSyscall(k).GetName()
            fmt.Fprintf(w, "%d\t%s\n", v, name)
        }
    }
    w.Flush()
}
 
func (s syscallCounter) getName(syscallID uint64) string {
    name, _ := seccomp.ScmpSyscall(syscallID).GetName()
    return name
}

最后結果:

Run:  [echo hello]
Wait err stop signal: trace/breakpoint trap
name: execve, id: 59
name: brk, id: 12
name: access, id: 21
name: mmap, id: 9
name: access, id: 21
name: open, id: 2
name: fstat, id: 5
name: mmap, id: 9
name: close, id: 3
name: access, id: 21
name: open, id: 2
name: read, id: 0
name: fstat, id: 5
name: mmap, id: 9
name: mprotect, id: 10
name: mmap, id: 9
name: mmap, id: 9
name: close, id: 3
name: mmap, id: 9
name: arch_prctl, id: 158
name: mprotect, id: 10
name: mprotect, id: 10
name: mprotect, id: 10
name: munmap, id: 11
name: brk, id: 12
name: brk, id: 12
name: open, id: 2
name: fstat, id: 5
name: mmap, id: 9
name: close, id: 3
name: fstat, id: 5
hello
name: write, id: 1
name: close, id: 3
name: close, id: 3
        1|read
        1|write
        3|open
        5|close
        4|fstat
        7|mmap
        4|mprotect
        1|munmap
        3|brk
        3|access
        1|execve
        1|arch_prctl

三、參考

Golang標準庫——syscall
參考URL: https://www.jianshu.com/p/44109d5e045b
Golang 與系統調用
參考URL: https://blog.csdn.net/weixin_33744141/article/details/89033990

以上就是Golang標準庫syscall詳解(什么是系統調用)的詳細內容,更多關于Golang標準庫syscall的資料請關注腳本之家其它相關文章!

您可能感興趣的文章:
  • 解決Golang 中使用WaitGroup的那點坑
  • 在golang中使用Sync.WaitGroup解決等待的問題
  • Golang中的sync包的WaitGroup操作
  • Golang中的sync.WaitGroup用法實例
  • Golang的os標準庫中常用函數的整理介紹
  • Golang 標準庫 tips之waitgroup詳解

標簽:宜賓 江西 張掖 黑龍江 延邊 嘉峪關 武漢 新余

巨人網絡通訊聲明:本文標題《Golang標準庫syscall詳解(什么是系統調用)》,本文關鍵詞  Golang,標準,庫,syscall,詳解,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Golang標準庫syscall詳解(什么是系統調用)》相關的同類信息!
  • 本頁收集關于Golang標準庫syscall詳解(什么是系統調用)的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    国产成人丝袜美腿| 成人午夜碰碰视频| 亚洲激情一二三区| 日韩欧美在线123| 蜜臀精品久久久久久蜜臀| 在线观看日韩电影| 天天影视涩香欲综合网| 国产精品盗摄一区二区三区| 91精品国产综合久久久蜜臀粉嫩 | 波多野结衣中文字幕一区二区三区 | 久久―日本道色综合久久| 99麻豆久久久国产精品免费 | 欧美欧美欧美欧美首页| 成人黄色电影在线| 青草av.久久免费一区| 亚洲女人的天堂| 日本伊人精品一区二区三区观看方式| 丝袜美腿亚洲综合| 亚洲一卡二卡三卡四卡无卡久久| 精品第一国产综合精品aⅴ| 国产伦精一区二区三区| 亚洲成在人线在线播放| 国产精品激情偷乱一区二区∴| 欧美做爰猛烈大尺度电影无法无天| 国产成人av电影在线播放| 久久久美女艺术照精彩视频福利播放| 久久人人爽人人爽| 国产精品亚洲а∨天堂免在线| 国产一区美女在线| 欧美性大战xxxxx久久久| 国产午夜精品一区二区三区视频 | 婷婷综合久久一区二区三区| 成人app软件下载大全免费| av资源网一区| 欧美丝袜丝交足nylons图片| 欧美一二三在线| 国产日韩av一区| 亚洲综合男人的天堂| 青娱乐精品视频在线| 国产精品99久久久| 91久久国产最好的精华液| 欧美成人官网二区| 一区二区三区在线免费观看| 极品少妇xxxx精品少妇| 91麻豆国产香蕉久久精品| 欧美一区二区三区在线视频| 国产欧美日韩视频一区二区| 亚洲一区二区三区四区五区中文 | 国产精品99久久久久久久vr| 欧美在线免费观看亚洲| 久久久美女毛片| 五月天激情综合网| 99re8在线精品视频免费播放| 日韩精品自拍偷拍| 亚洲男同性恋视频| 国产老妇另类xxxxx| 欧美日本乱大交xxxxx| 国产精品区一区二区三| 蜜臀av在线播放一区二区三区| 成人免费电影视频| 精品国产一二三区| 亚洲电影中文字幕在线观看| 国产电影一区在线| 日韩一二三区视频| 偷拍与自拍一区| 91国偷自产一区二区开放时间| 久久人人爽人人爽| 激情图区综合网| 日韩亚洲欧美在线观看| 亚洲国产综合人成综合网站| 99久久精品国产观看| 国产精品无码永久免费888| 九九九精品视频| 欧美大片一区二区| 麻豆久久久久久| 91精品中文字幕一区二区三区| 亚洲天堂福利av| 波多野结衣精品在线| 欧美高清在线一区| 成人午夜在线播放| 国产欧美精品一区二区色综合 | 国产精品久久久久久久久免费桃花 | 久久超碰97人人做人人爱| 欧美日韩极品在线观看一区| 亚洲午夜免费视频| 欧美午夜片在线看| 亚洲成人中文在线| 91麻豆精品国产91久久久久久久久| 亚洲一区二区不卡免费| 欧美日韩中文字幕一区二区| 亚洲成人av在线电影| 欧美日韩在线播放三区四区| 亚洲超碰精品一区二区| 91精品国产91久久久久久最新毛片| 精品一区二区在线视频| 国产清纯白嫩初高生在线观看91 | 日韩在线卡一卡二| 精品国内片67194| 成人黄色av电影| 亚洲在线成人精品| 精品入口麻豆88视频| 成人福利在线看| 亚洲一区二区四区蜜桃| 日韩欧美成人激情| 99精品在线观看视频| 亚洲精品福利视频网站| 欧美一区二区三区日韩| 丰满少妇久久久久久久| 亚洲一区二区中文在线| 日韩精品综合一本久道在线视频| 粉嫩av一区二区三区| 香蕉久久夜色精品国产使用方法| 欧美精品一区二区在线播放| 99riav久久精品riav| 日韩电影免费在线看| 国产精品成人一区二区三区夜夜夜| 欧美午夜寂寞影院| 国内外精品视频| 午夜一区二区三区视频| 国产区在线观看成人精品| 欧美日韩国产首页在线观看| 国产精品一区一区| 日本不卡一二三| 亚洲乱码国产乱码精品精小说 | 一区二区三区四区激情| 精品国产一区二区三区av性色 | 国产精品亚洲专一区二区三区| 一区二区三区四区蜜桃| 欧美国产一区二区| 欧美sm美女调教| 欧美日韩日日摸| 日本精品一区二区三区四区的功能| 国产专区欧美精品| 亚洲综合区在线| 亚洲色图19p| 1区2区3区欧美| 久久欧美一区二区| 日韩精品在线看片z| 欧美日韩成人综合天天影院| 在线精品视频一区二区三四| 不卡av电影在线播放| 粉嫩在线一区二区三区视频| 狠狠色综合色综合网络| 国产精品国产三级国产普通话三级 | 日韩欧美精品在线视频| 欧美视频精品在线观看| 色94色欧美sute亚洲线路二| 99久久综合狠狠综合久久| 国产成人精品www牛牛影视| 国产尤物一区二区在线| 奇米四色…亚洲| 日本欧美肥老太交大片| 老司机精品视频在线| 蜜桃av一区二区在线观看| 视频在线观看91| 日韩毛片在线免费观看| 综合久久国产九一剧情麻豆| 国产精品白丝在线| 亚洲欧洲另类国产综合| 亚洲精品视频一区| 亚洲永久免费视频| 亚洲国产精品一区二区www| 亚洲精品国产a久久久久久| 亚洲一区二区三区国产| 香蕉久久夜色精品国产使用方法| 亚洲va欧美va天堂v国产综合| 日本网站在线观看一区二区三区| 久久成人综合网| 波多野结衣中文字幕一区| 欧美视频三区在线播放| 日韩色在线观看| 日本一区二区综合亚洲| 亚洲人成7777| 日韩福利电影在线观看| 国内精品在线播放| 99免费精品视频| 欧美一区二区免费观在线| 精品国产露脸精彩对白| 中文字幕一区二区三| 亚洲国产精品影院| 国产一区二区精品久久99| 99久久99久久免费精品蜜臀| 91黄色小视频| 日韩欧美成人一区| 亚洲天堂a在线| 蜜臀va亚洲va欧美va天堂| 99久久亚洲一区二区三区青草| 欧美亚洲一区三区| 国产人伦精品一区二区| 日韩av中文在线观看| 国产成a人无v码亚洲福利| 欧美图区在线视频| 中文字幕国产一区二区| 午夜精品视频一区| 成人h动漫精品一区二区| 欧美一级欧美三级在线观看| 国产精品毛片久久久久久| 日本中文字幕一区二区视频 | 亚洲最色的网站| 精品中文字幕一区二区小辣椒|