1
0
out-of-tree/debug.go

224 lines
4.6 KiB
Go
Raw Normal View History

2018-11-25 14:46:29 +00:00
// Copyright 2018 Mikhail Klementev. All rights reserved.
// Use of this source code is governed by a AGPLv3 license
// (or later) that can be found in the LICENSE file.
package main
import (
"errors"
"fmt"
"io/ioutil"
"log"
"os"
"strings"
"time"
"gopkg.in/logrusorgru/aurora.v2"
2019-02-02 21:24:29 +00:00
"code.dumpstack.io/tools/out-of-tree/config"
"code.dumpstack.io/tools/out-of-tree/qemu"
2018-11-25 14:46:29 +00:00
)
func firstSupported(kcfg config.KernelConfig, ka config.Artifact,
kernel string) (ki config.KernelInfo, err error) {
km, err := kernelMask(kernel)
if err != nil {
return
}
ka.SupportedKernels = []config.KernelMask{km}
for _, ki = range kcfg.Kernels {
var supported bool
supported, err = ka.Supported(ki)
if err != nil || supported {
return
}
}
err = errors.New("No supported kernel found")
return
}
2019-08-17 09:05:06 +00:00
func handleLine(q *qemu.System) (err error) {
2018-11-25 14:46:29 +00:00
fmt.Print("out-of-tree> ")
rawLine := "help"
fmt.Scanf("%s", &rawLine)
params := strings.Fields(rawLine)
cmd := params[0]
switch cmd {
case "h", "help":
fmt.Printf("help\t: print this help message\n")
fmt.Printf("log\t: print qemu log\n")
fmt.Printf("clog\t: print qemu log and cleanup buffer\n")
fmt.Printf("cleanup\t: cleanup qemu log buffer\n")
fmt.Printf("ssh\t: print arguments to ssh command\n")
fmt.Printf("quit\t: quit\n")
case "l", "log":
fmt.Println(string(q.Stdout))
case "cl", "clog":
fmt.Println(string(q.Stdout))
q.Stdout = []byte{}
case "c", "cleanup":
q.Stdout = []byte{}
case "s", "ssh":
2019-08-17 09:05:06 +00:00
fmt.Println(q.GetSSHCommand())
2018-11-25 14:46:29 +00:00
case "q", "quit":
return errors.New("end of session")
default:
fmt.Println("No such command")
}
return
}
2019-08-17 09:05:06 +00:00
func interactive(q *qemu.System) (err error) {
2018-11-25 14:46:29 +00:00
for {
err = handleLine(q)
if err != nil {
return
}
}
}
2018-11-25 14:53:56 +00:00
func debugHandler(kcfg config.KernelConfig, workPath, kernRegex, gdb string,
2019-08-20 00:05:19 +00:00
dockerTimeout time.Duration, yekaslr, yesmep, yesmap, yekpti,
nokaslr, nosmep, nosmap, nokpti bool) (err error) {
2018-11-25 14:46:29 +00:00
ka, err := config.ReadArtifactConfig(workPath + "/.out-of-tree.toml")
if err != nil {
return
}
if ka.SourcePath == "" {
ka.SourcePath = workPath
}
ki, err := firstSupported(kcfg, ka, kernRegex)
if err != nil {
return
}
kernel := qemu.Kernel{KernelPath: ki.KernelPath, InitrdPath: ki.InitrdPath}
2019-08-17 09:05:06 +00:00
q, err := qemu.NewSystem(qemu.X86x64, kernel, ki.RootFS)
2018-11-25 14:46:29 +00:00
if err != nil {
return
}
if ka.Qemu.Cpus != 0 {
q.Cpus = ka.Qemu.Cpus
}
if ka.Qemu.Memory != 0 {
q.Memory = ka.Qemu.Memory
}
if ka.Docker.Timeout.Duration != 0 {
dockerTimeout = ka.Docker.Timeout.Duration
}
q.SetKASLR(false) // set KASLR to false by default because of gdb
q.SetSMEP(!ka.Mitigations.DisableSmep)
q.SetSMAP(!ka.Mitigations.DisableSmap)
2019-08-20 00:05:19 +00:00
q.SetKPTI(!ka.Mitigations.DisableKpti)
if yekaslr {
q.SetKASLR(true)
} else if nokaslr {
q.SetKASLR(false)
}
if yesmep {
q.SetSMEP(true)
} else if nosmep {
q.SetSMEP(false)
}
if yesmap {
q.SetSMAP(true)
} else if nosmap {
q.SetSMAP(false)
}
2019-08-20 00:05:19 +00:00
if yekpti {
q.SetKPTI(true)
} else if nokpti {
q.SetKPTI(false)
}
redgreen := func(name string, enabled bool) aurora.Value {
if enabled {
return aurora.BgGreen(aurora.Black(name))
}
return aurora.BgRed(aurora.White(name))
}
2019-08-20 00:05:19 +00:00
fmt.Printf("[*] %s %s %s %s\n",
redgreen("KASLR", q.GetKASLR()),
redgreen("SMEP", q.GetSMEP()),
2019-08-20 00:05:19 +00:00
redgreen("SMAP", q.GetSMAP()),
redgreen("KPTI", q.GetKPTI()))
fmt.Printf("[*] SMP: %d CPUs\n", q.Cpus)
fmt.Printf("[*] Memory: %d MB\n", q.Memory)
2018-11-25 14:53:56 +00:00
q.Debug(gdb)
2018-11-25 14:46:29 +00:00
coloredGdbAddress := aurora.BgGreen(aurora.Black(gdb))
2019-08-21 06:16:25 +00:00
fmt.Printf("[*] gdb is listening on %s\n", coloredGdbAddress)
2018-11-25 14:46:29 +00:00
err = q.Start()
if err != nil {
return
}
defer q.Stop()
tmp, err := ioutil.TempDir("/tmp/", "out-of-tree_")
if err != nil {
return
}
defer os.RemoveAll(tmp)
2020-06-14 20:14:59 +00:00
err = preloadModules(q, ka, ki, dockerTimeout)
if err != nil {
log.Println(err)
return
}
2018-11-25 14:46:29 +00:00
outFile, output, err := build(tmp, ka, ki, dockerTimeout)
if err != nil {
log.Println(err, output)
return
}
2019-08-16 05:38:25 +00:00
remoteFile := "/tmp/exploit"
2018-11-25 14:46:29 +00:00
if ka.Type == config.KernelModule {
2019-08-16 05:38:25 +00:00
remoteFile = "/tmp/module.ko"
2018-11-25 14:46:29 +00:00
}
err = q.CopyFile("user", outFile, remoteFile)
if err != nil {
return
}
// Copy all test files to the remote machine
for _,f := range ka.TestFiles {
err = q.CopyFile(f.User,f.Local,f.Remote)
if err != nil {
log.Println("error copy err:", err, f.Local, f.Remote)
return
}
}
2018-11-25 14:46:29 +00:00
coloredRemoteFile := aurora.BgGreen(aurora.Black(remoteFile))
fmt.Printf("[*] build result copied to %s\n", coloredRemoteFile)
2019-08-17 09:05:06 +00:00
fmt.Printf("\n%s\n", q.GetSSHCommand())
fmt.Printf("gdb %s -ex 'target remote %s'\n\n", ki.VmlinuxPath, gdb)
2019-08-16 06:32:26 +00:00
// TODO set substitute-path /build/.../linux-... /path/to/linux-source
2018-11-25 14:46:29 +00:00
err = interactive(q)
return
}