| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115 |
- // Copyright The OpenTelemetry Authors
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- package ptrace
- import (
- "encoding/binary"
- "syscall"
- "github.com/pkg/errors"
- )
- const syscallInstrSize = 2
- func getIP(regs *syscall.PtraceRegs) uintptr {
- return uintptr(regs.Rip)
- }
- func getRegs(pid int, regsout *syscall.PtraceRegs) error {
- err := syscall.PtraceGetRegs(pid, regsout)
- if err != nil {
- return errors.Wrapf(err, "get registers of process %d", pid)
- }
- return nil
- }
- func setRegs(pid int, regs *syscall.PtraceRegs) error {
- err := syscall.PtraceSetRegs(pid, regs)
- if err != nil {
- return errors.Wrapf(err, "set registers of process %d", pid)
- }
- return nil
- }
- // Syscall runs a syscall at main thread of process.
- func (p *TracedProgram) Syscall(number uint64, args ...uint64) (uint64, error) {
- // save the original registers and the current instructions
- err := p.Protect()
- if err != nil {
- return 0, err
- }
- var regs syscall.PtraceRegs
- err = getRegs(p.pid, ®s)
- if err != nil {
- return 0, err
- }
- // set the registers according to the syscall convention. Learn more about
- // it in `man 2 syscall`. In x86_64 the syscall nr is stored in rax
- // register, and the arguments are stored in rdi, rsi, rdx, r10, r8, r9 in
- // order
- regs.Rax = number
- for index, arg := range args {
- // All these registers are hard coded for x86 platform
- if index == 0 {
- regs.Rdi = arg
- } else if index == 1 {
- regs.Rsi = arg
- } else if index == 2 {
- regs.Rdx = arg
- } else if index == 3 {
- regs.R10 = arg
- } else if index == 4 {
- regs.R8 = arg
- } else if index == 5 {
- regs.R9 = arg
- } else {
- return 0, errors.New("too many arguments for a syscall")
- }
- }
- err = setRegs(p.pid, ®s)
- if err != nil {
- return 0, err
- }
- instruction := make([]byte, syscallInstrSize)
- ip := getIP(p.backupRegs)
- // set the current instruction (the ip register points to) to the `syscall`
- // instruction. In x86_64, the `syscall` instruction is 0x050f.
- binary.LittleEndian.PutUint16(instruction, 0x050f)
- _, err = syscall.PtracePokeData(p.pid, ip, instruction)
- if err != nil {
- return 0, errors.Wrapf(err, "writing data %v to %x", instruction, ip)
- }
- // run one instruction, and stop
- err = p.Step()
- if err != nil {
- return 0, err
- }
- // read registers, the return value of syscall is stored inside rax register
- err = getRegs(p.pid, ®s)
- if err != nil {
- return 0, err
- }
- // restore the state saved at beginning.
- return regs.Rax, p.Restore()
- }
|