mirror of
https://github.com/beego/bee.git
synced 2024-12-22 15:10:49 +00:00
commit
e6bb3f893c
16
.travis.yml
16
.travis.yml
@ -1,5 +1,19 @@
|
|||||||
language: go
|
language: go
|
||||||
|
|
||||||
go:
|
go:
|
||||||
- 1.6.3
|
- 1.6.3
|
||||||
- 1.7.3
|
- 1.7.3
|
||||||
|
- 1.8
|
||||||
|
install:
|
||||||
|
- export PATH=$PATH:$HOME/gopath/bin
|
||||||
|
- go get -u github.com/opennota/check/cmd/structcheck
|
||||||
|
- go get -u honnef.co/go/tools/cmd/gosimple
|
||||||
|
- go get -u honnef.co/go/tools/cmd/staticcheck
|
||||||
|
- go get -u honnef.co/go/tools/cmd/unused
|
||||||
|
- go get -u github.com/mdempsky/unconvert
|
||||||
|
script:
|
||||||
|
- go vet $(go list ./... | grep -v /vendor/)
|
||||||
|
- structcheck $(go list ./... | grep -v /vendor/)
|
||||||
|
- gosimple -ignore "github.com/beego/bee/cmd/commands/run/*.go:S1024" $(go list ./... | grep -v /vendor/)
|
||||||
|
- staticcheck $(go list ./... | grep -v /vendor/)
|
||||||
|
- unused $(go list ./... | grep -v /vendor/)
|
||||||
|
- unconvert $(go list ./... | grep -v /vendor/)
|
||||||
|
3
Beefile
3
Beefile
@ -1,7 +1,4 @@
|
|||||||
version: 0
|
version: 0
|
||||||
gopm:
|
|
||||||
enable: false
|
|
||||||
install: false
|
|
||||||
go_install: false
|
go_install: false
|
||||||
watch_ext: []
|
watch_ext: []
|
||||||
dir_structure:
|
dir_structure:
|
||||||
|
81
README.md
81
README.md
@ -1,3 +1,4 @@
|
|||||||
|
|
||||||
bee
|
bee
|
||||||
===
|
===
|
||||||
|
|
||||||
@ -34,17 +35,20 @@ go get -u github.com/beego/bee
|
|||||||
Bee provides a variety of commands which can be helpful at various stages of development. The top level commands include:
|
Bee provides a variety of commands which can be helpful at various stages of development. The top level commands include:
|
||||||
|
|
||||||
```
|
```
|
||||||
new Creates a Beego application
|
|
||||||
run Run the application by starting a local development server
|
|
||||||
pack Compresses a Beego application into a single file
|
|
||||||
api Creates a Beego API application
|
|
||||||
hprose Creates an RPC application based on Hprose and Beego frameworks
|
|
||||||
bale Transforms non-Go files to Go source files
|
|
||||||
version Prints the current Bee version
|
version Prints the current Bee version
|
||||||
generate Source code generator
|
|
||||||
migrate Runs database migrations
|
migrate Runs database migrations
|
||||||
|
api Creates a Beego API application
|
||||||
|
bale Transforms non-Go files to Go source files
|
||||||
fix Fixes your application by making it compatible with newer versions of Beego
|
fix Fixes your application by making it compatible with newer versions of Beego
|
||||||
|
dlv Start a debugging session using Delve
|
||||||
dockerize Generates a Dockerfile for your Beego application
|
dockerize Generates a Dockerfile for your Beego application
|
||||||
|
generate Source code generator
|
||||||
|
hprose Creates an RPC application based on Hprose and Beego frameworks
|
||||||
|
new Creates a Beego application
|
||||||
|
pack Compresses a Beego application into a single file
|
||||||
|
rs Run customized scripts
|
||||||
|
run Run the application by starting a local development server
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### bee version
|
### bee version
|
||||||
@ -165,6 +169,38 @@ ______
|
|||||||
|
|
||||||
For more information on the usage, run `bee help pack`.
|
For more information on the usage, run `bee help pack`.
|
||||||
|
|
||||||
|
### bee rs
|
||||||
|
Inspired by makefile / npm scripts.
|
||||||
|
Run script allows you to run arbitrary commands using Bee.
|
||||||
|
Custom commands are provided from the "scripts" object inside bee.json or Beefile.
|
||||||
|
|
||||||
|
To run a custom command, use: $ bee rs mycmd ARGS
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ bee help rs
|
||||||
|
|
||||||
|
USAGE
|
||||||
|
bee rs
|
||||||
|
|
||||||
|
DESCRIPTION
|
||||||
|
Run script allows you to run arbitrary commands using Bee.
|
||||||
|
Custom commands are provided from the "scripts" object inside bee.json or Beefile.
|
||||||
|
|
||||||
|
To run a custom command, use: $ bee rs mycmd ARGS
|
||||||
|
|
||||||
|
AVAILABLE SCRIPTS
|
||||||
|
gtest
|
||||||
|
APP_ENV=test APP_CONF_PATH=$(pwd)/conf go test -v -cover
|
||||||
|
gtestall
|
||||||
|
APP_ENV=test APP_CONF_PATH=$(pwd)/conf go test -v -cover $(go list ./... | grep -v /vendor/)
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
*Run your scripts with:*
|
||||||
|
```$ bee rs gtest tests/*.go```
|
||||||
|
```$ bee rs gtestall```
|
||||||
|
|
||||||
|
|
||||||
### bee api
|
### bee api
|
||||||
|
|
||||||
To create a Beego API application:
|
To create a Beego API application:
|
||||||
@ -298,6 +334,37 @@ ______
|
|||||||
|
|
||||||
For more information on the usage, run `bee help dockerize`.
|
For more information on the usage, run `bee help dockerize`.
|
||||||
|
|
||||||
|
### bee dlv
|
||||||
|
|
||||||
|
Bee can also help with debugging your application. To start a debugging session:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
______
|
||||||
|
| ___ \
|
||||||
|
| |_/ / ___ ___
|
||||||
|
| ___ \ / _ \ / _ \
|
||||||
|
| |_/ /| __/| __/
|
||||||
|
\____/ \___| \___| v1.8.0
|
||||||
|
2017/03/22 11:17:05 INFO ▶ 0001 Starting Delve Debugger...
|
||||||
|
Type 'help' for list of commands.
|
||||||
|
(dlv) break main.main
|
||||||
|
Breakpoint 1 set at 0x40100f for main.main() ./main.go:8
|
||||||
|
|
||||||
|
(dlv) continue
|
||||||
|
> main.main() ./main.go:8 (hits goroutine(1):1 total:1) (PC: 0x40100f)
|
||||||
|
3: import (
|
||||||
|
4: _ "github.com/user/myapp/routers"
|
||||||
|
5: "github.com/astaxie/beego"
|
||||||
|
6: )
|
||||||
|
7:
|
||||||
|
=> 8: func main() {
|
||||||
|
9: beego.Run()
|
||||||
|
10: }
|
||||||
|
11:
|
||||||
|
```
|
||||||
|
|
||||||
|
For more information on the usage, run `bee help dlv`.
|
||||||
|
|
||||||
## Shortcuts
|
## Shortcuts
|
||||||
|
|
||||||
Because you'll likely type these generator commands over and over, it makes sense to create aliases:
|
Because you'll likely type these generator commands over and over, it makes sense to create aliases:
|
||||||
|
280
autorouter.go
280
autorouter.go
@ -1,280 +0,0 @@
|
|||||||
// Copyright 2013 bee 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 main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"errors"
|
|
||||||
"fmt"
|
|
||||||
"go/ast"
|
|
||||||
gobuild "go/build"
|
|
||||||
"go/doc"
|
|
||||||
"go/parser"
|
|
||||||
"go/token"
|
|
||||||
"io"
|
|
||||||
"io/ioutil"
|
|
||||||
"os"
|
|
||||||
"path"
|
|
||||||
"runtime"
|
|
||||||
"strings"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
var cmdRouter = &Command{
|
|
||||||
UsageLine: "router",
|
|
||||||
Short: "auto-generate routers for the app controllers",
|
|
||||||
Long: `
|
|
||||||
|
|
||||||
`,
|
|
||||||
}
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
cmdRouter.Run = autoRouter
|
|
||||||
}
|
|
||||||
|
|
||||||
func autoRouter(cmd *Command, args []string) int {
|
|
||||||
fmt.Println("[INFO] Starting auto-generating routers...")
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
// getControllerInfo returns controllers that embeded "beego.controller"
|
|
||||||
// and their methods of package in given path.
|
|
||||||
func getControllerInfo(path string) (map[string][]string, error) {
|
|
||||||
now := time.Now()
|
|
||||||
path = strings.TrimSuffix(path, "/")
|
|
||||||
dir, err := os.Open(path)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
fis, err := dir.Readdir(0)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
files := make([]*source, 0, len(fis))
|
|
||||||
for _, fi := range fis {
|
|
||||||
// Only load Go files
|
|
||||||
if strings.HasSuffix(fi.Name(), ".go") {
|
|
||||||
f, err := os.Open(path + "/" + fi.Name())
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
p := make([]byte, fi.Size())
|
|
||||||
_, err = f.Read(p)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
f.Close()
|
|
||||||
files = append(files, &source{
|
|
||||||
name: path + "/" + fi.Name(),
|
|
||||||
data: p,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
rw := &routerWalker{
|
|
||||||
pdoc: &Package{
|
|
||||||
ImportPath: path,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
cm := make(map[string][]string)
|
|
||||||
pdoc, err := rw.build(files)
|
|
||||||
for _, t := range pdoc.Types {
|
|
||||||
// Check if embeded "beego.Controller".
|
|
||||||
if strings.Index(t.Decl, "beego.Controller") > -1 {
|
|
||||||
for _, f := range t.Methods {
|
|
||||||
cm[t.Name] = append(cm[t.Name], f.Name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
fmt.Println(time.Since(now))
|
|
||||||
return cm, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// source represents a source code file.
|
|
||||||
type source struct {
|
|
||||||
name string
|
|
||||||
data []byte
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *source) Name() string { return s.name }
|
|
||||||
func (s *source) Size() int64 { return int64(len(s.data)) }
|
|
||||||
func (s *source) Mode() os.FileMode { return 0 }
|
|
||||||
func (s *source) ModTime() time.Time { return time.Time{} }
|
|
||||||
func (s *source) IsDir() bool { return false }
|
|
||||||
func (s *source) Sys() interface{} { return nil }
|
|
||||||
|
|
||||||
// routerWalker holds the state used when building the documentation.
|
|
||||||
type routerWalker struct {
|
|
||||||
pdoc *Package
|
|
||||||
srcs map[string]*source // Source files
|
|
||||||
fset *token.FileSet
|
|
||||||
buf []byte // scratch space for printNode method
|
|
||||||
}
|
|
||||||
|
|
||||||
// Package represents full information and documentation for a package.
|
|
||||||
type Package struct {
|
|
||||||
ImportPath string
|
|
||||||
Types []*Type // Top-level declarations
|
|
||||||
}
|
|
||||||
|
|
||||||
// Type represents structs and interfaces.
|
|
||||||
type Type struct {
|
|
||||||
Name string // Type name
|
|
||||||
Decl string
|
|
||||||
Methods []*Func // Exported methods
|
|
||||||
}
|
|
||||||
|
|
||||||
// Func represents functions
|
|
||||||
type Func struct {
|
|
||||||
Name string
|
|
||||||
}
|
|
||||||
|
|
||||||
// build generates data from source files.
|
|
||||||
func (w *routerWalker) build(srcs []*source) (*Package, error) {
|
|
||||||
// Add source files to walker, I skipped references here
|
|
||||||
w.srcs = make(map[string]*source)
|
|
||||||
for _, src := range srcs {
|
|
||||||
w.srcs[src.name] = src
|
|
||||||
}
|
|
||||||
|
|
||||||
w.fset = token.NewFileSet()
|
|
||||||
|
|
||||||
// Find the package and associated files
|
|
||||||
ctxt := gobuild.Context{
|
|
||||||
GOOS: runtime.GOOS,
|
|
||||||
GOARCH: runtime.GOARCH,
|
|
||||||
CgoEnabled: true,
|
|
||||||
JoinPath: path.Join,
|
|
||||||
IsAbsPath: path.IsAbs,
|
|
||||||
SplitPathList: func(list string) []string { return strings.Split(list, ":") },
|
|
||||||
IsDir: func(path string) bool { panic("unexpected") },
|
|
||||||
HasSubdir: func(root, dir string) (rel string, ok bool) { panic("unexpected") },
|
|
||||||
ReadDir: func(dir string) (fi []os.FileInfo, err error) { return w.readDir(dir) },
|
|
||||||
OpenFile: func(path string) (r io.ReadCloser, err error) { return w.openFile(path) },
|
|
||||||
Compiler: "gc",
|
|
||||||
}
|
|
||||||
|
|
||||||
bpkg, err := ctxt.ImportDir(w.pdoc.ImportPath, 0)
|
|
||||||
// Continue if there are no Go source files; we still want the directory info
|
|
||||||
_, nogo := err.(*gobuild.NoGoError)
|
|
||||||
if err != nil {
|
|
||||||
if nogo {
|
|
||||||
err = nil
|
|
||||||
} else {
|
|
||||||
return nil, errors.New("routerWalker.build -> " + err.Error())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse the Go files
|
|
||||||
files := make(map[string]*ast.File)
|
|
||||||
for _, name := range append(bpkg.GoFiles, bpkg.CgoFiles...) {
|
|
||||||
file, err := parser.ParseFile(w.fset, name, w.srcs[name].data, parser.ParseComments)
|
|
||||||
if err != nil {
|
|
||||||
return nil, errors.New("routerWalker.build -> parse go files: " + err.Error())
|
|
||||||
}
|
|
||||||
files[name] = file
|
|
||||||
}
|
|
||||||
|
|
||||||
apkg, _ := ast.NewPackage(w.fset, files, simpleImporter, nil)
|
|
||||||
|
|
||||||
mode := doc.Mode(0)
|
|
||||||
if w.pdoc.ImportPath == "builtin" {
|
|
||||||
mode |= doc.AllDecls
|
|
||||||
}
|
|
||||||
|
|
||||||
pdoc := doc.New(apkg, w.pdoc.ImportPath, mode)
|
|
||||||
|
|
||||||
w.pdoc.Types = w.types(pdoc.Types)
|
|
||||||
|
|
||||||
return w.pdoc, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (w *routerWalker) funcs(fdocs []*doc.Func) []*Func {
|
|
||||||
var result []*Func
|
|
||||||
for _, d := range fdocs {
|
|
||||||
result = append(result, &Func{
|
|
||||||
Name: d.Name,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return result
|
|
||||||
}
|
|
||||||
|
|
||||||
func (w *routerWalker) types(tdocs []*doc.Type) []*Type {
|
|
||||||
var result []*Type
|
|
||||||
for _, d := range tdocs {
|
|
||||||
result = append(result, &Type{
|
|
||||||
Decl: w.printDecl(d.Decl),
|
|
||||||
Name: d.Name,
|
|
||||||
Methods: w.funcs(d.Methods),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return result
|
|
||||||
}
|
|
||||||
|
|
||||||
func (w *routerWalker) printDecl(decl ast.Node) string {
|
|
||||||
var d Code
|
|
||||||
d, w.buf = printDecl(decl, w.fset, w.buf)
|
|
||||||
return d.Text
|
|
||||||
}
|
|
||||||
|
|
||||||
func (w *routerWalker) readDir(dir string) ([]os.FileInfo, error) {
|
|
||||||
if dir != w.pdoc.ImportPath {
|
|
||||||
panic("unexpected")
|
|
||||||
}
|
|
||||||
fis := make([]os.FileInfo, 0, len(w.srcs))
|
|
||||||
for _, src := range w.srcs {
|
|
||||||
fis = append(fis, src)
|
|
||||||
}
|
|
||||||
return fis, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (w *routerWalker) openFile(path string) (io.ReadCloser, error) {
|
|
||||||
if strings.HasPrefix(path, w.pdoc.ImportPath+"/") {
|
|
||||||
if src, ok := w.srcs[path[len(w.pdoc.ImportPath)+1:]]; ok {
|
|
||||||
return ioutil.NopCloser(bytes.NewReader(src.data)), nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
panic("unexpected")
|
|
||||||
}
|
|
||||||
|
|
||||||
func simpleImporter(imports map[string]*ast.Object, path string) (*ast.Object, error) {
|
|
||||||
pkg := imports[path]
|
|
||||||
if pkg == nil {
|
|
||||||
// Guess the package name without importing it. Start with the last
|
|
||||||
// element of the path.
|
|
||||||
name := path[strings.LastIndex(path, "/")+1:]
|
|
||||||
|
|
||||||
// Trim commonly used prefixes and suffixes containing illegal name
|
|
||||||
// runes.
|
|
||||||
name = strings.TrimSuffix(name, ".go")
|
|
||||||
name = strings.TrimSuffix(name, "-go")
|
|
||||||
name = strings.TrimPrefix(name, "go.")
|
|
||||||
name = strings.TrimPrefix(name, "go-")
|
|
||||||
name = strings.TrimPrefix(name, "biogo.")
|
|
||||||
|
|
||||||
// It's also common for the last element of the path to contain an
|
|
||||||
// extra "go" prefix, but not always.
|
|
||||||
// TODO: examine unresolved ids to detect when trimming the "go" prefix is appropriate.
|
|
||||||
pkg = ast.NewObj(ast.Pkg, name)
|
|
||||||
pkg.Data = ast.NewScope(nil)
|
|
||||||
imports[path] = pkg
|
|
||||||
}
|
|
||||||
return pkg, nil
|
|
||||||
}
|
|
@ -1,9 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestGetControllerInfo(t *testing.T) {
|
|
||||||
getControllerInfo("testdata/router/")
|
|
||||||
}
|
|
258
bee.go
258
bee.go
@ -1,258 +0,0 @@
|
|||||||
// Copyright 2013 bee 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.
|
|
||||||
|
|
||||||
// Bee is a tool for developing applications based on beego framework.
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"flag"
|
|
||||||
"fmt"
|
|
||||||
"io"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
"strings"
|
|
||||||
"text/template"
|
|
||||||
)
|
|
||||||
|
|
||||||
const version = "1.8.0"
|
|
||||||
|
|
||||||
// Command is the unit of execution
|
|
||||||
type Command struct {
|
|
||||||
// Run runs the command.
|
|
||||||
// The args are the arguments after the command name.
|
|
||||||
Run func(cmd *Command, args []string) int
|
|
||||||
|
|
||||||
// PreRun performs an operation before running the command
|
|
||||||
PreRun func(cmd *Command, args []string)
|
|
||||||
|
|
||||||
// UsageLine is the one-line usage message.
|
|
||||||
// The first word in the line is taken to be the command name.
|
|
||||||
UsageLine string
|
|
||||||
|
|
||||||
// Short is the short description shown in the 'go help' output.
|
|
||||||
Short string
|
|
||||||
|
|
||||||
// Long is the long message shown in the 'go help <this-command>' output.
|
|
||||||
Long string
|
|
||||||
|
|
||||||
// Flag is a set of flags specific to this command.
|
|
||||||
Flag flag.FlagSet
|
|
||||||
|
|
||||||
// CustomFlags indicates that the command will do its own
|
|
||||||
// flag parsing.
|
|
||||||
CustomFlags bool
|
|
||||||
|
|
||||||
// output out writer if set in SetOutput(w)
|
|
||||||
output *io.Writer
|
|
||||||
}
|
|
||||||
|
|
||||||
// Name returns the command's name: the first word in the usage line.
|
|
||||||
func (c *Command) Name() string {
|
|
||||||
name := c.UsageLine
|
|
||||||
i := strings.Index(name, " ")
|
|
||||||
if i >= 0 {
|
|
||||||
name = name[:i]
|
|
||||||
}
|
|
||||||
return name
|
|
||||||
}
|
|
||||||
|
|
||||||
// SetOutput sets the destination for usage and error messages.
|
|
||||||
// If output is nil, os.Stderr is used.
|
|
||||||
func (c *Command) SetOutput(output io.Writer) {
|
|
||||||
c.output = &output
|
|
||||||
}
|
|
||||||
|
|
||||||
// Out returns the out writer of the current command.
|
|
||||||
// If cmd.output is nil, os.Stderr is used.
|
|
||||||
func (c *Command) Out() io.Writer {
|
|
||||||
if c.output != nil {
|
|
||||||
return *c.output
|
|
||||||
}
|
|
||||||
return NewColorWriter(os.Stderr)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Usage puts out the usage for the command.
|
|
||||||
func (c *Command) Usage() {
|
|
||||||
tmpl(cmdUsage, c)
|
|
||||||
os.Exit(2)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Runnable reports whether the command can be run; otherwise
|
|
||||||
// it is a documentation pseudo-command such as import path.
|
|
||||||
func (c *Command) Runnable() bool {
|
|
||||||
return c.Run != nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Command) Options() map[string]string {
|
|
||||||
options := make(map[string]string)
|
|
||||||
c.Flag.VisitAll(func(f *flag.Flag) {
|
|
||||||
defaultVal := f.DefValue
|
|
||||||
if len(defaultVal) > 0 {
|
|
||||||
options[f.Name+"="+defaultVal] = f.Usage
|
|
||||||
} else {
|
|
||||||
options[f.Name] = f.Usage
|
|
||||||
}
|
|
||||||
})
|
|
||||||
return options
|
|
||||||
}
|
|
||||||
|
|
||||||
var availableCommands = []*Command{
|
|
||||||
cmdNew,
|
|
||||||
cmdRun,
|
|
||||||
cmdPack,
|
|
||||||
cmdApiapp,
|
|
||||||
cmdHproseapp,
|
|
||||||
//cmdRouter,
|
|
||||||
//cmdTest,
|
|
||||||
cmdBale,
|
|
||||||
cmdVersion,
|
|
||||||
cmdGenerate,
|
|
||||||
//cmdRundocs,
|
|
||||||
cmdMigrate,
|
|
||||||
cmdFix,
|
|
||||||
cmdDockerize,
|
|
||||||
}
|
|
||||||
|
|
||||||
var logger = GetBeeLogger(os.Stdout)
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
currentpath, _ := os.Getwd()
|
|
||||||
|
|
||||||
flag.Usage = usage
|
|
||||||
flag.Parse()
|
|
||||||
log.SetFlags(0)
|
|
||||||
|
|
||||||
args := flag.Args()
|
|
||||||
|
|
||||||
if len(args) < 1 {
|
|
||||||
usage()
|
|
||||||
}
|
|
||||||
|
|
||||||
if args[0] == "help" {
|
|
||||||
help(args[1:])
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, cmd := range availableCommands {
|
|
||||||
if cmd.Name() == args[0] && cmd.Run != nil {
|
|
||||||
cmd.Flag.Usage = func() { cmd.Usage() }
|
|
||||||
if cmd.CustomFlags {
|
|
||||||
args = args[1:]
|
|
||||||
} else {
|
|
||||||
cmd.Flag.Parse(args[1:])
|
|
||||||
args = cmd.Flag.Args()
|
|
||||||
}
|
|
||||||
|
|
||||||
if cmd.PreRun != nil {
|
|
||||||
cmd.PreRun(cmd, args)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if current directory is inside the GOPATH,
|
|
||||||
// if so parse the packages inside it.
|
|
||||||
if IsInGOPATH(currentpath) && isGenerateDocs(cmd.Name(), args) {
|
|
||||||
parsePackagesFromDir(currentpath)
|
|
||||||
}
|
|
||||||
|
|
||||||
os.Exit(cmd.Run(cmd, args))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
printErrorAndExit("Unknown subcommand")
|
|
||||||
}
|
|
||||||
|
|
||||||
func isGenerateDocs(name string, args []string) bool {
|
|
||||||
if name != "generate" {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
for _, a := range args {
|
|
||||||
if a == "docs" {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
var usageTemplate = `Bee is a Fast and Flexible tool for managing your Beego Web Application.
|
|
||||||
|
|
||||||
{{"USAGE" | headline}}
|
|
||||||
{{"bee command [arguments]" | bold}}
|
|
||||||
|
|
||||||
{{"AVAILABLE COMMANDS" | headline}}
|
|
||||||
{{range .}}{{if .Runnable}}
|
|
||||||
{{.Name | printf "%-11s" | bold}} {{.Short}}{{end}}{{end}}
|
|
||||||
|
|
||||||
Use {{"bee help [command]" | bold}} for more information about a command.
|
|
||||||
|
|
||||||
{{"ADDITIONAL HELP TOPICS" | headline}}
|
|
||||||
{{range .}}{{if not .Runnable}}
|
|
||||||
{{.Name | printf "%-11s"}} {{.Short}}{{end}}{{end}}
|
|
||||||
|
|
||||||
Use {{"bee help [topic]" | bold}} for more information about that topic.
|
|
||||||
`
|
|
||||||
|
|
||||||
var helpTemplate = `{{"USAGE" | headline}}
|
|
||||||
{{.UsageLine | printf "bee %s" | bold}}
|
|
||||||
{{if .Options}}{{endline}}{{"OPTIONS" | headline}}{{range $k,$v := .Options}}
|
|
||||||
{{$k | printf "-%s" | bold}}
|
|
||||||
{{$v}}
|
|
||||||
{{end}}{{end}}
|
|
||||||
{{"DESCRIPTION" | headline}}
|
|
||||||
{{tmpltostr .Long . | trim}}
|
|
||||||
`
|
|
||||||
|
|
||||||
var errorTemplate = `bee: %s.
|
|
||||||
Use {{"bee help" | bold}} for more information.
|
|
||||||
`
|
|
||||||
|
|
||||||
var cmdUsage = `Use {{printf "bee help %s" .Name | bold}} for more information.{{endline}}`
|
|
||||||
|
|
||||||
func usage() {
|
|
||||||
tmpl(usageTemplate, availableCommands)
|
|
||||||
os.Exit(2)
|
|
||||||
}
|
|
||||||
|
|
||||||
func tmpl(text string, data interface{}) {
|
|
||||||
output := NewColorWriter(os.Stderr)
|
|
||||||
|
|
||||||
t := template.New("usage").Funcs(BeeFuncMap())
|
|
||||||
template.Must(t.Parse(text))
|
|
||||||
|
|
||||||
err := t.Execute(output, data)
|
|
||||||
MustCheck(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
func help(args []string) {
|
|
||||||
if len(args) == 0 {
|
|
||||||
usage()
|
|
||||||
}
|
|
||||||
if len(args) != 1 {
|
|
||||||
printErrorAndExit("Too many arguments")
|
|
||||||
}
|
|
||||||
|
|
||||||
arg := args[0]
|
|
||||||
|
|
||||||
for _, cmd := range availableCommands {
|
|
||||||
if cmd.Name() == arg {
|
|
||||||
tmpl(helpTemplate, cmd)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
printErrorAndExit("Unknown help topic")
|
|
||||||
}
|
|
||||||
|
|
||||||
func printErrorAndExit(message string) {
|
|
||||||
tmpl(fmt.Sprintf(errorTemplate, message), nil)
|
|
||||||
os.Exit(2)
|
|
||||||
}
|
|
4
bee.json
4
bee.json
@ -1,9 +1,5 @@
|
|||||||
{
|
{
|
||||||
"version": 0,
|
"version": 0,
|
||||||
"gopm": {
|
|
||||||
"enable": false,
|
|
||||||
"install": false
|
|
||||||
},
|
|
||||||
"go_install": false,
|
"go_install": false,
|
||||||
"watch_ext": [],
|
"watch_ext": [],
|
||||||
"dir_structure": {
|
"dir_structure": {
|
||||||
|
101
cmd/bee.go
Normal file
101
cmd/bee.go
Normal file
@ -0,0 +1,101 @@
|
|||||||
|
// Copyright 2013 bee 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 cmd ...
|
||||||
|
package cmd
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
_ "github.com/beego/bee/cmd/commands/api"
|
||||||
|
_ "github.com/beego/bee/cmd/commands/bale"
|
||||||
|
_ "github.com/beego/bee/cmd/commands/beefix"
|
||||||
|
_ "github.com/beego/bee/cmd/commands/dlv"
|
||||||
|
_ "github.com/beego/bee/cmd/commands/dockerize"
|
||||||
|
_ "github.com/beego/bee/cmd/commands/generate"
|
||||||
|
_ "github.com/beego/bee/cmd/commands/hprose"
|
||||||
|
_ "github.com/beego/bee/cmd/commands/migrate"
|
||||||
|
_ "github.com/beego/bee/cmd/commands/new"
|
||||||
|
_ "github.com/beego/bee/cmd/commands/pack"
|
||||||
|
_ "github.com/beego/bee/cmd/commands/rs"
|
||||||
|
_ "github.com/beego/bee/cmd/commands/run"
|
||||||
|
_ "github.com/beego/bee/cmd/commands/version"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
|
)
|
||||||
|
|
||||||
|
func IfGenerateDocs(name string, args []string) bool {
|
||||||
|
if name != "generate" {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
for _, a := range args {
|
||||||
|
if a == "docs" {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
var usageTemplate = `Bee is a Fast and Flexible tool for managing your Beego Web Application.
|
||||||
|
|
||||||
|
{{"USAGE" | headline}}
|
||||||
|
{{"bee command [arguments]" | bold}}
|
||||||
|
|
||||||
|
{{"AVAILABLE COMMANDS" | headline}}
|
||||||
|
{{range .}}{{if .Runnable}}
|
||||||
|
{{.Name | printf "%-11s" | bold}} {{.Short}}{{end}}{{end}}
|
||||||
|
|
||||||
|
Use {{"bee help [command]" | bold}} for more information about a command.
|
||||||
|
|
||||||
|
{{"ADDITIONAL HELP TOPICS" | headline}}
|
||||||
|
{{range .}}{{if not .Runnable}}
|
||||||
|
{{.Name | printf "%-11s"}} {{.Short}}{{end}}{{end}}
|
||||||
|
|
||||||
|
Use {{"bee help [topic]" | bold}} for more information about that topic.
|
||||||
|
`
|
||||||
|
|
||||||
|
var helpTemplate = `{{"USAGE" | headline}}
|
||||||
|
{{.UsageLine | printf "bee %s" | bold}}
|
||||||
|
{{if .Options}}{{endline}}{{"OPTIONS" | headline}}{{range $k,$v := .Options}}
|
||||||
|
{{$k | printf "-%s" | bold}}
|
||||||
|
{{$v}}
|
||||||
|
{{end}}{{end}}
|
||||||
|
{{"DESCRIPTION" | headline}}
|
||||||
|
{{tmpltostr .Long . | trim}}
|
||||||
|
`
|
||||||
|
|
||||||
|
var ErrorTemplate = `bee: %s.
|
||||||
|
Use {{"bee help" | bold}} for more information.
|
||||||
|
`
|
||||||
|
|
||||||
|
func Usage() {
|
||||||
|
utils.Tmpl(usageTemplate, commands.AvailableCommands)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Help(args []string) {
|
||||||
|
if len(args) == 0 {
|
||||||
|
Usage()
|
||||||
|
}
|
||||||
|
if len(args) != 1 {
|
||||||
|
utils.PrintErrorAndExit("Too many arguments", ErrorTemplate)
|
||||||
|
}
|
||||||
|
|
||||||
|
arg := args[0]
|
||||||
|
|
||||||
|
for _, cmd := range commands.AvailableCommands {
|
||||||
|
if cmd.Name() == arg {
|
||||||
|
utils.Tmpl(helpTemplate, cmd)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
utils.PrintErrorAndExit("Unknown help topic", ErrorTemplate)
|
||||||
|
}
|
@ -12,16 +12,22 @@
|
|||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
|
||||||
package main
|
package apiapp
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
path "path/filepath"
|
path "path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
"github.com/beego/bee/cmd/commands/version"
|
||||||
|
"github.com/beego/bee/generate"
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
var cmdApiapp = &Command{
|
var CmdApiapp = &commands.Command{
|
||||||
// CustomFlags: true,
|
// CustomFlags: true,
|
||||||
UsageLine: "api [appname]",
|
UsageLine: "api [appname]",
|
||||||
Short: "Creates a Beego API application",
|
Short: "Creates a Beego API application",
|
||||||
@ -50,10 +56,9 @@ var cmdApiapp = &Command{
|
|||||||
└── object.go
|
└── object.go
|
||||||
└── user.go
|
└── user.go
|
||||||
`,
|
`,
|
||||||
PreRun: func(cmd *Command, args []string) { ShowShortVersionBanner() },
|
PreRun: func(cmd *commands.Command, args []string) { version.ShowShortVersionBanner() },
|
||||||
Run: createapi,
|
Run: createAPI,
|
||||||
}
|
}
|
||||||
|
|
||||||
var apiconf = `appname = {{.Appname}}
|
var apiconf = `appname = {{.Appname}}
|
||||||
httpport = 8080
|
httpport = 8080
|
||||||
runmode = dev
|
runmode = dev
|
||||||
@ -134,7 +139,7 @@ func init() {
|
|||||||
}
|
}
|
||||||
`
|
`
|
||||||
|
|
||||||
var apiModels = `package models
|
var APIModels = `package models
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
@ -189,7 +194,7 @@ func Delete(ObjectId string) {
|
|||||||
|
|
||||||
`
|
`
|
||||||
|
|
||||||
var apiModels2 = `package models
|
var APIModels2 = `package models
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
@ -532,133 +537,101 @@ func TestGet(t *testing.T) {
|
|||||||
`
|
`
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
cmdApiapp.Flag.Var(&tables, "tables", "List of table names separated by a comma.")
|
CmdApiapp.Flag.Var(&generate.Tables, "tables", "List of table names separated by a comma.")
|
||||||
cmdApiapp.Flag.Var(&driver, "driver", "Database driver. Either mysql, postgres or sqlite.")
|
CmdApiapp.Flag.Var(&generate.SQLDriver, "driver", "Database driver. Either mysql, postgres or sqlite.")
|
||||||
cmdApiapp.Flag.Var(&conn, "conn", "Connection string used by the driver to connect to a database instance.")
|
CmdApiapp.Flag.Var(&generate.SQLConn, "conn", "Connection string used by the driver to connect to a database instance.")
|
||||||
|
commands.AvailableCommands = append(commands.AvailableCommands, CmdApiapp)
|
||||||
}
|
}
|
||||||
|
|
||||||
func createapi(cmd *Command, args []string) int {
|
func createAPI(cmd *commands.Command, args []string) int {
|
||||||
output := cmd.Out()
|
output := cmd.Out()
|
||||||
|
|
||||||
if len(args) < 1 {
|
if len(args) < 1 {
|
||||||
logger.Fatal("Argument [appname] is missing")
|
beeLogger.Log.Fatal("Argument [appname] is missing")
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(args) > 1 {
|
if len(args) > 1 {
|
||||||
cmd.Flag.Parse(args[1:])
|
err := cmd.Flag.Parse(args[1:])
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Error(err.Error())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
apppath, packpath, err := checkEnv(args[0])
|
appPath, packPath, err := utils.CheckEnv(args[0])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("%s", err)
|
beeLogger.Log.Fatalf("%s", err)
|
||||||
}
|
}
|
||||||
if driver == "" {
|
if generate.SQLDriver == "" {
|
||||||
driver = "mysql"
|
generate.SQLDriver = "mysql"
|
||||||
}
|
|
||||||
if conn == "" {
|
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.Info("Creating API...")
|
beeLogger.Log.Info("Creating API...")
|
||||||
|
|
||||||
os.MkdirAll(apppath, 0755)
|
os.MkdirAll(appPath, 0755)
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", apppath, "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", appPath, "\x1b[0m")
|
||||||
os.Mkdir(path.Join(apppath, "conf"), 0755)
|
os.Mkdir(path.Join(appPath, "conf"), 0755)
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "conf"), "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(appPath, "conf"), "\x1b[0m")
|
||||||
os.Mkdir(path.Join(apppath, "controllers"), 0755)
|
os.Mkdir(path.Join(appPath, "controllers"), 0755)
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "controllers"), "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(appPath, "controllers"), "\x1b[0m")
|
||||||
os.Mkdir(path.Join(apppath, "tests"), 0755)
|
os.Mkdir(path.Join(appPath, "tests"), 0755)
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "tests"), "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(appPath, "tests"), "\x1b[0m")
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "conf", "app.conf"), "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(appPath, "conf", "app.conf"), "\x1b[0m")
|
||||||
WriteToFile(path.Join(apppath, "conf", "app.conf"),
|
utils.WriteToFile(path.Join(appPath, "conf", "app.conf"),
|
||||||
strings.Replace(apiconf, "{{.Appname}}", path.Base(args[0]), -1))
|
strings.Replace(apiconf, "{{.Appname}}", path.Base(args[0]), -1))
|
||||||
|
|
||||||
if conn != "" {
|
if generate.SQLConn != "" {
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "main.go"), "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(appPath, "main.go"), "\x1b[0m")
|
||||||
maingoContent := strings.Replace(apiMainconngo, "{{.Appname}}", packpath, -1)
|
mainGoContent := strings.Replace(apiMainconngo, "{{.Appname}}", packPath, -1)
|
||||||
maingoContent = strings.Replace(maingoContent, "{{.DriverName}}", string(driver), -1)
|
mainGoContent = strings.Replace(mainGoContent, "{{.DriverName}}", string(generate.SQLDriver), -1)
|
||||||
if driver == "mysql" {
|
if generate.SQLDriver == "mysql" {
|
||||||
maingoContent = strings.Replace(maingoContent, "{{.DriverPkg}}", `_ "github.com/go-sql-driver/mysql"`, -1)
|
mainGoContent = strings.Replace(mainGoContent, "{{.DriverPkg}}", `_ "github.com/go-sql-driver/mysql"`, -1)
|
||||||
} else if driver == "postgres" {
|
} else if generate.SQLDriver == "postgres" {
|
||||||
maingoContent = strings.Replace(maingoContent, "{{.DriverPkg}}", `_ "github.com/lib/pq"`, -1)
|
mainGoContent = strings.Replace(mainGoContent, "{{.DriverPkg}}", `_ "github.com/lib/pq"`, -1)
|
||||||
}
|
}
|
||||||
WriteToFile(path.Join(apppath, "main.go"),
|
utils.WriteToFile(path.Join(appPath, "main.go"),
|
||||||
strings.Replace(
|
strings.Replace(
|
||||||
maingoContent,
|
mainGoContent,
|
||||||
"{{.conn}}",
|
"{{.conn}}",
|
||||||
conn.String(),
|
generate.SQLConn.String(),
|
||||||
-1,
|
-1,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
logger.Infof("Using '%s' as 'driver'", driver)
|
beeLogger.Log.Infof("Using '%s' as 'driver'", generate.SQLDriver)
|
||||||
logger.Infof("Using '%s' as 'conn'", conn)
|
beeLogger.Log.Infof("Using '%s' as 'conn'", generate.SQLConn)
|
||||||
logger.Infof("Using '%s' as 'tables'", tables)
|
beeLogger.Log.Infof("Using '%s' as 'tables'", generate.Tables)
|
||||||
generateAppcode(string(driver), string(conn), "3", string(tables), apppath)
|
generate.GenerateAppcode(string(generate.SQLDriver), string(generate.SQLConn), "3", string(generate.Tables), appPath)
|
||||||
} else {
|
} else {
|
||||||
os.Mkdir(path.Join(apppath, "models"), 0755)
|
os.Mkdir(path.Join(appPath, "models"), 0755)
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "models"), "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(appPath, "models"), "\x1b[0m")
|
||||||
os.Mkdir(path.Join(apppath, "routers"), 0755)
|
os.Mkdir(path.Join(appPath, "routers"), 0755)
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "routers")+string(path.Separator), "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(appPath, "routers")+string(path.Separator), "\x1b[0m")
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "controllers", "object.go"), "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(appPath, "controllers", "object.go"), "\x1b[0m")
|
||||||
WriteToFile(path.Join(apppath, "controllers", "object.go"),
|
utils.WriteToFile(path.Join(appPath, "controllers", "object.go"),
|
||||||
strings.Replace(apiControllers, "{{.Appname}}", packpath, -1))
|
strings.Replace(apiControllers, "{{.Appname}}", packPath, -1))
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "controllers", "user.go"), "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(appPath, "controllers", "user.go"), "\x1b[0m")
|
||||||
WriteToFile(path.Join(apppath, "controllers", "user.go"),
|
utils.WriteToFile(path.Join(appPath, "controllers", "user.go"),
|
||||||
strings.Replace(apiControllers2, "{{.Appname}}", packpath, -1))
|
strings.Replace(apiControllers2, "{{.Appname}}", packPath, -1))
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "tests", "default_test.go"), "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(appPath, "tests", "default_test.go"), "\x1b[0m")
|
||||||
WriteToFile(path.Join(apppath, "tests", "default_test.go"),
|
utils.WriteToFile(path.Join(appPath, "tests", "default_test.go"),
|
||||||
strings.Replace(apiTests, "{{.Appname}}", packpath, -1))
|
strings.Replace(apiTests, "{{.Appname}}", packPath, -1))
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "routers", "router.go"), "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(appPath, "routers", "router.go"), "\x1b[0m")
|
||||||
WriteToFile(path.Join(apppath, "routers", "router.go"),
|
utils.WriteToFile(path.Join(appPath, "routers", "router.go"),
|
||||||
strings.Replace(apirouter, "{{.Appname}}", packpath, -1))
|
strings.Replace(apirouter, "{{.Appname}}", packPath, -1))
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "models", "object.go"), "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(appPath, "models", "object.go"), "\x1b[0m")
|
||||||
WriteToFile(path.Join(apppath, "models", "object.go"), apiModels)
|
utils.WriteToFile(path.Join(appPath, "models", "object.go"), APIModels)
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "models", "user.go"), "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(appPath, "models", "user.go"), "\x1b[0m")
|
||||||
WriteToFile(path.Join(apppath, "models", "user.go"), apiModels2)
|
utils.WriteToFile(path.Join(appPath, "models", "user.go"), APIModels2)
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "main.go"), "\x1b[0m")
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(appPath, "main.go"), "\x1b[0m")
|
||||||
WriteToFile(path.Join(apppath, "main.go"),
|
utils.WriteToFile(path.Join(appPath, "main.go"),
|
||||||
strings.Replace(apiMaingo, "{{.Appname}}", packpath, -1))
|
strings.Replace(apiMaingo, "{{.Appname}}", packPath, -1))
|
||||||
}
|
}
|
||||||
logger.Success("New API successfully created!")
|
beeLogger.Log.Success("New API successfully created!")
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
func checkEnv(appname string) (apppath, packpath string, err error) {
|
|
||||||
gps := GetGOPATHs()
|
|
||||||
if len(gps) == 0 {
|
|
||||||
logger.Fatal("GOPATH environment variable is not set or empty")
|
|
||||||
}
|
|
||||||
currpath, _ := os.Getwd()
|
|
||||||
currpath = path.Join(currpath, appname)
|
|
||||||
for _, gpath := range gps {
|
|
||||||
gsrcpath := path.Join(gpath, "src")
|
|
||||||
if strings.HasPrefix(currpath, gsrcpath) {
|
|
||||||
packpath = strings.Replace(currpath[len(gsrcpath)+1:], string(path.Separator), "/", -1)
|
|
||||||
return currpath, packpath, nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// In case of multiple paths in the GOPATH, by default
|
|
||||||
// we use the first path
|
|
||||||
gopath := gps[0]
|
|
||||||
|
|
||||||
logger.Warn("You current workdir is not inside $GOPATH/src.")
|
|
||||||
logger.Debugf("GOPATH: %s", __FILE__(), __LINE__(), gopath)
|
|
||||||
|
|
||||||
gosrcpath := path.Join(gopath, "src")
|
|
||||||
apppath = path.Join(gosrcpath, appname)
|
|
||||||
|
|
||||||
if _, e := os.Stat(apppath); os.IsNotExist(e) == false {
|
|
||||||
err = fmt.Errorf("Cannot create application without removing '%s' first.", apppath)
|
|
||||||
logger.Errorf("Path '%s' already exists", apppath)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
packpath = strings.Join(strings.Split(apppath[len(gosrcpath)+1:], string(path.Separator)), "/")
|
|
||||||
return
|
|
||||||
}
|
|
@ -11,8 +11,7 @@
|
|||||||
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
package bale
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
@ -24,9 +23,15 @@ import (
|
|||||||
"path/filepath"
|
"path/filepath"
|
||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
"github.com/beego/bee/cmd/commands/version"
|
||||||
|
"github.com/beego/bee/config"
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
var cmdBale = &Command{
|
var CmdBale = &commands.Command{
|
||||||
UsageLine: "bale",
|
UsageLine: "bale",
|
||||||
Short: "Transforms non-Go files to Go source files",
|
Short: "Transforms non-Go files to Go source files",
|
||||||
Long: `Bale command compress all the static files in to a single binary file.
|
Long: `Bale command compress all the static files in to a single binary file.
|
||||||
@ -37,47 +42,46 @@ var cmdBale = &Command{
|
|||||||
It will auto-generate an unpack function to the main package then run it during the runtime.
|
It will auto-generate an unpack function to the main package then run it during the runtime.
|
||||||
This is mainly used for zealots who are requiring 100% Go code.
|
This is mainly used for zealots who are requiring 100% Go code.
|
||||||
`,
|
`,
|
||||||
PreRun: func(cmd *Command, args []string) { ShowShortVersionBanner() },
|
PreRun: func(cmd *commands.Command, args []string) { version.ShowShortVersionBanner() },
|
||||||
Run: runBale,
|
Run: runBale,
|
||||||
}
|
}
|
||||||
|
|
||||||
func runBale(cmd *Command, args []string) int {
|
func init() {
|
||||||
err := loadConfig()
|
commands.AvailableCommands = append(commands.AvailableCommands, CmdBale)
|
||||||
if err != nil {
|
}
|
||||||
logger.Fatalf("Failed to load configuration: %s", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
|
func runBale(cmd *commands.Command, args []string) int {
|
||||||
os.RemoveAll("bale")
|
os.RemoveAll("bale")
|
||||||
os.Mkdir("bale", os.ModePerm)
|
os.Mkdir("bale", os.ModePerm)
|
||||||
|
|
||||||
// Pack and compress data
|
// Pack and compress data
|
||||||
for _, p := range conf.Bale.Dirs {
|
for _, p := range config.Conf.Bale.Dirs {
|
||||||
if !isExist(p) {
|
if !utils.IsExist(p) {
|
||||||
logger.Warnf("Skipped directory: %s", p)
|
beeLogger.Log.Warnf("Skipped directory: %s", p)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
logger.Infof("Packaging directory: %s", p)
|
beeLogger.Log.Infof("Packaging directory: %s", p)
|
||||||
filepath.Walk(p, walkFn)
|
filepath.Walk(p, walkFn)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Generate auto-uncompress function.
|
// Generate auto-uncompress function.
|
||||||
buf := new(bytes.Buffer)
|
buf := new(bytes.Buffer)
|
||||||
buf.WriteString(fmt.Sprintf(BaleHeader, conf.Bale.Import,
|
buf.WriteString(fmt.Sprintf(BaleHeader, config.Conf.Bale.Import,
|
||||||
strings.Join(resFiles, "\",\n\t\t\""),
|
strings.Join(resFiles, "\",\n\t\t\""),
|
||||||
strings.Join(resFiles, ",\n\t\tbale.R")))
|
strings.Join(resFiles, ",\n\t\tbale.R")))
|
||||||
|
|
||||||
fw, err := os.Create("bale.go")
|
fw, err := os.Create("bale.go")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Failed to create file: %s", err)
|
beeLogger.Log.Fatalf("Failed to create file: %s", err)
|
||||||
}
|
}
|
||||||
defer fw.Close()
|
defer fw.Close()
|
||||||
|
|
||||||
_, err = fw.Write(buf.Bytes())
|
_, err = fw.Write(buf.Bytes())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Failed to write data: %s", err)
|
beeLogger.Log.Fatalf("Failed to write data: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.Success("Baled resources successfully!")
|
beeLogger.Log.Success("Baled resources successfully!")
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -138,7 +142,7 @@ func saveFile(filePath string, b []byte) (int, error) {
|
|||||||
|
|
||||||
var resFiles = make([]string, 0, 10)
|
var resFiles = make([]string, 0, 10)
|
||||||
|
|
||||||
func walkFn(resPath string, info os.FileInfo, err error) error {
|
func walkFn(resPath string, info os.FileInfo, _ error) error {
|
||||||
if info.IsDir() || filterSuffix(resPath) {
|
if info.IsDir() || filterSuffix(resPath) {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -146,7 +150,7 @@ func walkFn(resPath string, info os.FileInfo, err error) error {
|
|||||||
// Open resource files
|
// Open resource files
|
||||||
fr, err := os.Open(resPath)
|
fr, err := os.Open(resPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Failed to read file: %s", err)
|
beeLogger.Log.Fatalf("Failed to read file: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Convert path
|
// Convert path
|
||||||
@ -164,7 +168,7 @@ func walkFn(resPath string, info os.FileInfo, err error) error {
|
|||||||
os.MkdirAll(path.Dir(resPath), os.ModePerm)
|
os.MkdirAll(path.Dir(resPath), os.ModePerm)
|
||||||
fw, err := os.Create("bale/" + resPath + ".go")
|
fw, err := os.Create("bale/" + resPath + ".go")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Failed to create file: %s", err)
|
beeLogger.Log.Fatalf("Failed to create file: %s", err)
|
||||||
}
|
}
|
||||||
defer fw.Close()
|
defer fw.Close()
|
||||||
|
|
||||||
@ -184,7 +188,7 @@ func walkFn(resPath string, info os.FileInfo, err error) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func filterSuffix(name string) bool {
|
func filterSuffix(name string) bool {
|
||||||
for _, s := range conf.Bale.IngExt {
|
for _, s := range config.Conf.Bale.IngExt {
|
||||||
if strings.HasSuffix(name, s) {
|
if strings.HasSuffix(name, s) {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
@ -1,4 +1,4 @@
|
|||||||
package main
|
package beefix
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
@ -9,9 +9,14 @@ import (
|
|||||||
"path/filepath"
|
"path/filepath"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
"github.com/beego/bee/cmd/commands/version"
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/logger/colors"
|
||||||
)
|
)
|
||||||
|
|
||||||
var cmdFix = &Command{
|
var CmdFix = &commands.Command{
|
||||||
UsageLine: "fix",
|
UsageLine: "fix",
|
||||||
Short: "Fixes your application by making it compatible with newer versions of Beego",
|
Short: "Fixes your application by making it compatible with newer versions of Beego",
|
||||||
Long: `As of {{"Beego 1.6"|bold}}, there are some backward compatibility issues.
|
Long: `As of {{"Beego 1.6"|bold}}, there are some backward compatibility issues.
|
||||||
@ -22,18 +27,19 @@ var cmdFix = &Command{
|
|||||||
}
|
}
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
cmdFix.Run = runFix
|
CmdFix.Run = runFix
|
||||||
cmdFix.PreRun = func(cmd *Command, args []string) { ShowShortVersionBanner() }
|
CmdFix.PreRun = func(cmd *commands.Command, args []string) { version.ShowShortVersionBanner() }
|
||||||
|
commands.AvailableCommands = append(commands.AvailableCommands, CmdFix)
|
||||||
}
|
}
|
||||||
|
|
||||||
func runFix(cmd *Command, args []string) int {
|
func runFix(cmd *commands.Command, args []string) int {
|
||||||
output := cmd.Out()
|
output := cmd.Out()
|
||||||
|
|
||||||
logger.Info("Upgrading the application...")
|
beeLogger.Log.Info("Upgrading the application...")
|
||||||
|
|
||||||
dir, err := os.Getwd()
|
dir, err := os.Getwd()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Error while getting the current working directory: %s", err)
|
beeLogger.Log.Fatalf("Error while getting the current working directory: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
filepath.Walk(dir, func(path string, info os.FileInfo, err error) error {
|
filepath.Walk(dir, func(path string, info os.FileInfo, err error) error {
|
||||||
@ -50,13 +56,13 @@ func runFix(cmd *Command, args []string) int {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
err = fixFile(path)
|
err = fixFile(path)
|
||||||
fmt.Fprintf(output, GreenBold("\tfix\t")+"%s\n", path)
|
fmt.Fprintf(output, colors.GreenBold("\tfix\t")+"%s\n", path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Errorf("Could not fix file: %s", err)
|
beeLogger.Log.Errorf("Could not fix file: %s", err)
|
||||||
}
|
}
|
||||||
return err
|
return err
|
||||||
})
|
})
|
||||||
logger.Success("Upgrade Done!")
|
beeLogger.Log.Success("Upgrade Done!")
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
94
cmd/commands/command.go
Normal file
94
cmd/commands/command.go
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
package commands
|
||||||
|
|
||||||
|
import (
|
||||||
|
"flag"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/beego/bee/logger/colors"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Command is the unit of execution
|
||||||
|
type Command struct {
|
||||||
|
// Run runs the command.
|
||||||
|
// The args are the arguments after the command name.
|
||||||
|
Run func(cmd *Command, args []string) int
|
||||||
|
|
||||||
|
// PreRun performs an operation before running the command
|
||||||
|
PreRun func(cmd *Command, args []string)
|
||||||
|
|
||||||
|
// UsageLine is the one-line Usage message.
|
||||||
|
// The first word in the line is taken to be the command name.
|
||||||
|
UsageLine string
|
||||||
|
|
||||||
|
// Short is the short description shown in the 'go help' output.
|
||||||
|
Short string
|
||||||
|
|
||||||
|
// Long is the long message shown in the 'go help <this-command>' output.
|
||||||
|
Long string
|
||||||
|
|
||||||
|
// Flag is a set of flags specific to this command.
|
||||||
|
Flag flag.FlagSet
|
||||||
|
|
||||||
|
// CustomFlags indicates that the command will do its own
|
||||||
|
// flag parsing.
|
||||||
|
CustomFlags bool
|
||||||
|
|
||||||
|
// output out writer if set in SetOutput(w)
|
||||||
|
output *io.Writer
|
||||||
|
}
|
||||||
|
|
||||||
|
var AvailableCommands = []*Command{}
|
||||||
|
var cmdUsage = `Use {{printf "bee help %s" .Name | bold}} for more information.{{endline}}`
|
||||||
|
|
||||||
|
// Name returns the command's name: the first word in the Usage line.
|
||||||
|
func (c *Command) Name() string {
|
||||||
|
name := c.UsageLine
|
||||||
|
i := strings.Index(name, " ")
|
||||||
|
if i >= 0 {
|
||||||
|
name = name[:i]
|
||||||
|
}
|
||||||
|
return name
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetOutput sets the destination for Usage and error messages.
|
||||||
|
// If output is nil, os.Stderr is used.
|
||||||
|
func (c *Command) SetOutput(output io.Writer) {
|
||||||
|
c.output = &output
|
||||||
|
}
|
||||||
|
|
||||||
|
// Out returns the out writer of the current command.
|
||||||
|
// If cmd.output is nil, os.Stderr is used.
|
||||||
|
func (c *Command) Out() io.Writer {
|
||||||
|
if c.output != nil {
|
||||||
|
return *c.output
|
||||||
|
}
|
||||||
|
return colors.NewColorWriter(os.Stderr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Usage puts out the Usage for the command.
|
||||||
|
func (c *Command) Usage() {
|
||||||
|
utils.Tmpl(cmdUsage, c)
|
||||||
|
os.Exit(2)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Runnable reports whether the command can be run; otherwise
|
||||||
|
// it is a documentation pseudo-command such as import path.
|
||||||
|
func (c *Command) Runnable() bool {
|
||||||
|
return c.Run != nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Command) Options() map[string]string {
|
||||||
|
options := make(map[string]string)
|
||||||
|
c.Flag.VisitAll(func(f *flag.Flag) {
|
||||||
|
defaultVal := f.DefValue
|
||||||
|
if len(defaultVal) > 0 {
|
||||||
|
options[f.Name+"="+defaultVal] = f.Usage
|
||||||
|
} else {
|
||||||
|
options[f.Name] = f.Usage
|
||||||
|
}
|
||||||
|
})
|
||||||
|
return options
|
||||||
|
}
|
135
cmd/commands/dlv/dlv.go
Normal file
135
cmd/commands/dlv/dlv.go
Normal file
@ -0,0 +1,135 @@
|
|||||||
|
// Copyright 2017 bee 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 dlv ...
|
||||||
|
package dlv
|
||||||
|
|
||||||
|
import (
|
||||||
|
"flag"
|
||||||
|
"fmt"
|
||||||
|
"net"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
"github.com/beego/bee/cmd/commands/version"
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
|
"github.com/derekparker/delve/pkg/terminal"
|
||||||
|
"github.com/derekparker/delve/service"
|
||||||
|
"github.com/derekparker/delve/service/rpc2"
|
||||||
|
"github.com/derekparker/delve/service/rpccommon"
|
||||||
|
)
|
||||||
|
|
||||||
|
var cmdDlv = &commands.Command{
|
||||||
|
CustomFlags: true,
|
||||||
|
UsageLine: "dlv [-package=\"\"] [-port=8181]",
|
||||||
|
Short: "Start a debugging session using Delve",
|
||||||
|
Long: `dlv command start a debugging session using debugging tool Delve.
|
||||||
|
|
||||||
|
To debug your application using Delve, use: {{"$ bee dlv" | bold}}
|
||||||
|
|
||||||
|
For more information on Delve: https://github.com/derekparker/delve
|
||||||
|
`,
|
||||||
|
PreRun: func(cmd *commands.Command, args []string) { version.ShowShortVersionBanner() },
|
||||||
|
Run: runDlv,
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
packageName string
|
||||||
|
port int
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
fs := flag.NewFlagSet("dlv", flag.ContinueOnError)
|
||||||
|
fs.IntVar(&port, "port", 8181, "Port to listen to for clients")
|
||||||
|
fs.StringVar(&packageName, "package", "", "The package to debug (Must have a main package)")
|
||||||
|
cmdDlv.Flag = *fs
|
||||||
|
commands.AvailableCommands = append(commands.AvailableCommands, cmdDlv)
|
||||||
|
}
|
||||||
|
|
||||||
|
func runDlv(cmd *commands.Command, args []string) int {
|
||||||
|
if err := cmd.Flag.Parse(args); err != nil {
|
||||||
|
beeLogger.Log.Fatalf("Error parsing flags: %v", err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
debugname := "debug"
|
||||||
|
addr := fmt.Sprintf("127.0.0.1:%d", port)
|
||||||
|
return runDelve(addr, debugname)
|
||||||
|
}
|
||||||
|
|
||||||
|
// runDelve runs the Delve debugger server
|
||||||
|
func runDelve(addr, debugname string) int {
|
||||||
|
beeLogger.Log.Info("Starting Delve Debugger...")
|
||||||
|
|
||||||
|
err := gobuild(debugname, packageName)
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Fatalf("%v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
fp, err := filepath.Abs("./" + debugname)
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Fatalf("%v", err)
|
||||||
|
}
|
||||||
|
defer os.Remove(fp)
|
||||||
|
|
||||||
|
abs, err := filepath.Abs(debugname)
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Fatalf("%v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create and start the debugger server
|
||||||
|
listener, err := net.Listen("tcp", addr)
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Fatalf("Could not start listener: %s", err)
|
||||||
|
}
|
||||||
|
defer listener.Close()
|
||||||
|
|
||||||
|
server := rpccommon.NewServer(&service.Config{
|
||||||
|
Listener: listener,
|
||||||
|
AcceptMulti: true,
|
||||||
|
AttachPid: 0,
|
||||||
|
APIVersion: 2,
|
||||||
|
WorkingDir: "./",
|
||||||
|
ProcessArgs: []string{abs},
|
||||||
|
}, false)
|
||||||
|
if err := server.Run(); err != nil {
|
||||||
|
beeLogger.Log.Fatalf("Could not start debugger server: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start the Delve client REPL
|
||||||
|
client := rpc2.NewClient(addr)
|
||||||
|
term := terminal.New(client, nil)
|
||||||
|
|
||||||
|
status, err := term.Run()
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Fatalf("Could not start Delve REPL: %v", err)
|
||||||
|
}
|
||||||
|
defer term.Close()
|
||||||
|
|
||||||
|
// Stop and kill the debugger server once
|
||||||
|
// user quits the REPL
|
||||||
|
if err := server.Stop(true); err != nil {
|
||||||
|
beeLogger.Log.Fatalf("Could not stop Delve server: %v", err)
|
||||||
|
}
|
||||||
|
return status
|
||||||
|
}
|
||||||
|
|
||||||
|
// gobuild runs the "go build" command on the specified package
|
||||||
|
func gobuild(debugname, pkg string) error {
|
||||||
|
args := []string{"-gcflags", "-N -l", "-o", debugname}
|
||||||
|
args = append(args, utils.SplitQuotedFields("-ldflags='-linkmode internal'")...)
|
||||||
|
args = append(args, pkg)
|
||||||
|
return utils.GoCommand("build", args...)
|
||||||
|
}
|
@ -12,7 +12,7 @@
|
|||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
|
||||||
package main
|
package dockerize
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"flag"
|
"flag"
|
||||||
@ -21,22 +21,13 @@ import (
|
|||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
"text/template"
|
"text/template"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
"github.com/beego/bee/cmd/commands/version"
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
var cmdDockerize = &Command{
|
|
||||||
CustomFlags: true,
|
|
||||||
UsageLine: "dockerize",
|
|
||||||
Short: "Generates a Dockerfile for your Beego application",
|
|
||||||
Long: `Dockerize generates a Dockerfile for your Beego Web Application.
|
|
||||||
The Dockerfile will compile, get the dependencies with {{"godep"|bold}}, and set the entrypoint.
|
|
||||||
|
|
||||||
{{"Example:"|bold}}
|
|
||||||
$ bee dockerize -expose="3000,80,25"
|
|
||||||
`,
|
|
||||||
PreRun: func(cmd *Command, args []string) { ShowShortVersionBanner() },
|
|
||||||
Run: dockerizeApp,
|
|
||||||
}
|
|
||||||
|
|
||||||
const dockerBuildTemplate = `FROM {{.BaseImage}}
|
const dockerBuildTemplate = `FROM {{.BaseImage}}
|
||||||
|
|
||||||
# Godep for vendoring
|
# Godep for vendoring
|
||||||
@ -49,12 +40,11 @@ ENV APP_DIR $GOPATH{{.Appdir}}
|
|||||||
RUN mkdir -p $APP_DIR
|
RUN mkdir -p $APP_DIR
|
||||||
|
|
||||||
# Set the entrypoint
|
# Set the entrypoint
|
||||||
ENTRYPOINT $APP_DIR/{{.Entrypoint}}
|
ENTRYPOINT (cd $APP_DIR && ./{{.Entrypoint}})
|
||||||
ADD . $APP_DIR
|
ADD . $APP_DIR
|
||||||
|
|
||||||
# Compile the binary and statically link
|
# Compile the binary and statically link
|
||||||
RUN cd $APP_DIR
|
RUN cd $APP_DIR && CGO_ENABLED=0 godep go build -ldflags '-d -w -s'
|
||||||
RUN CGO_ENABLED=0 godep go build -ldflags '-d -w -s'
|
|
||||||
|
|
||||||
EXPOSE {{.Expose}}
|
EXPOSE {{.Expose}}
|
||||||
`
|
`
|
||||||
@ -67,6 +57,20 @@ type Dockerfile struct {
|
|||||||
Expose string
|
Expose string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var CmdDockerize = &commands.Command{
|
||||||
|
CustomFlags: true,
|
||||||
|
UsageLine: "dockerize",
|
||||||
|
Short: "Generates a Dockerfile for your Beego application",
|
||||||
|
Long: `Dockerize generates a Dockerfile for your Beego Web Application.
|
||||||
|
The Dockerfile will compile, get the dependencies with {{"godep"|bold}}, and set the entrypoint.
|
||||||
|
|
||||||
|
{{"Example:"|bold}}
|
||||||
|
$ bee dockerize -expose="3000,80,25"
|
||||||
|
`,
|
||||||
|
PreRun: func(cmd *commands.Command, args []string) { version.ShowShortVersionBanner() },
|
||||||
|
Run: dockerizeApp,
|
||||||
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
expose string
|
expose string
|
||||||
baseImage string
|
baseImage string
|
||||||
@ -76,19 +80,22 @@ func init() {
|
|||||||
fs := flag.NewFlagSet("dockerize", flag.ContinueOnError)
|
fs := flag.NewFlagSet("dockerize", flag.ContinueOnError)
|
||||||
fs.StringVar(&baseImage, "image", "library/golang", "Set the base image of the Docker container.")
|
fs.StringVar(&baseImage, "image", "library/golang", "Set the base image of the Docker container.")
|
||||||
fs.StringVar(&expose, "expose", "8080", "Port(s) to expose in the Docker container.")
|
fs.StringVar(&expose, "expose", "8080", "Port(s) to expose in the Docker container.")
|
||||||
cmdDockerize.Flag = *fs
|
CmdDockerize.Flag = *fs
|
||||||
|
commands.AvailableCommands = append(commands.AvailableCommands, CmdDockerize)
|
||||||
}
|
}
|
||||||
|
|
||||||
func dockerizeApp(cmd *Command, args []string) int {
|
func dockerizeApp(cmd *commands.Command, args []string) int {
|
||||||
if err := cmd.Flag.Parse(args); err != nil {
|
if err := cmd.Flag.Parse(args); err != nil {
|
||||||
logger.Fatalf("Error parsing flags: %v", err.Error())
|
beeLogger.Log.Fatalf("Error parsing flags: %v", err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.Info("Generating Dockerfile...")
|
beeLogger.Log.Info("Generating Dockerfile...")
|
||||||
|
|
||||||
gopath := os.Getenv("GOPATH")
|
gopath := os.Getenv("GOPATH")
|
||||||
dir, err := filepath.Abs(".")
|
dir, err := filepath.Abs(".")
|
||||||
MustCheck(err)
|
if err != nil {
|
||||||
|
beeLogger.Log.Error(err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
appdir := strings.Replace(dir, gopath, "", 1)
|
appdir := strings.Replace(dir, gopath, "", 1)
|
||||||
|
|
||||||
@ -112,15 +119,15 @@ func dockerizeApp(cmd *Command, args []string) int {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func generateDockerfile(df Dockerfile) {
|
func generateDockerfile(df Dockerfile) {
|
||||||
t := template.Must(template.New("dockerBuildTemplate").Parse(dockerBuildTemplate)).Funcs(BeeFuncMap())
|
t := template.Must(template.New("dockerBuildTemplate").Parse(dockerBuildTemplate)).Funcs(utils.BeeFuncMap())
|
||||||
|
|
||||||
f, err := os.Create("Dockerfile")
|
f, err := os.Create("Dockerfile")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Error writing Dockerfile: %v", err.Error())
|
beeLogger.Log.Fatalf("Error writing Dockerfile: %v", err.Error())
|
||||||
}
|
}
|
||||||
defer CloseFile(f)
|
defer utils.CloseFile(f)
|
||||||
|
|
||||||
t.Execute(f, df)
|
t.Execute(f, df)
|
||||||
|
|
||||||
logger.Success("Dockerfile generated.")
|
beeLogger.Log.Success("Dockerfile generated.")
|
||||||
}
|
}
|
216
cmd/commands/generate/generate.go
Normal file
216
cmd/commands/generate/generate.go
Normal file
@ -0,0 +1,216 @@
|
|||||||
|
// Copyright 2013 bee 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 generate
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
"github.com/beego/bee/cmd/commands/version"
|
||||||
|
"github.com/beego/bee/config"
|
||||||
|
"github.com/beego/bee/generate"
|
||||||
|
"github.com/beego/bee/generate/swaggergen"
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
|
)
|
||||||
|
|
||||||
|
var CmdGenerate = &commands.Command{
|
||||||
|
UsageLine: "generate [command]",
|
||||||
|
Short: "Source code generator",
|
||||||
|
Long: `▶ {{"To scaffold out your entire application:"|bold}}
|
||||||
|
|
||||||
|
$ bee generate scaffold [scaffoldname] [-fields="title:string,body:text"] [-driver=mysql] [-conn="root:@tcp(127.0.0.1:3306)/test"]
|
||||||
|
|
||||||
|
▶ {{"To generate a Model based on fields:"|bold}}
|
||||||
|
|
||||||
|
$ bee generate model [modelname] [-fields="name:type"]
|
||||||
|
|
||||||
|
▶ {{"To generate a controller:"|bold}}
|
||||||
|
|
||||||
|
$ bee generate controller [controllerfile]
|
||||||
|
|
||||||
|
▶ {{"To generate a CRUD view:"|bold}}
|
||||||
|
|
||||||
|
$ bee generate view [viewpath]
|
||||||
|
|
||||||
|
▶ {{"To generate a migration file for making database schema updates:"|bold}}
|
||||||
|
|
||||||
|
$ bee generate migration [migrationfile] [-fields="name:type"]
|
||||||
|
|
||||||
|
▶ {{"To generate swagger doc file:"|bold}}
|
||||||
|
|
||||||
|
$ bee generate docs
|
||||||
|
|
||||||
|
▶ {{"To generate a test case:"|bold}}
|
||||||
|
|
||||||
|
$ bee generate test [routerfile]
|
||||||
|
|
||||||
|
▶ {{"To generate appcode based on an existing database:"|bold}}
|
||||||
|
|
||||||
|
$ bee generate appcode [-tables=""] [-driver=mysql] [-conn="root:@tcp(127.0.0.1:3306)/test"] [-level=3]
|
||||||
|
`,
|
||||||
|
PreRun: func(cmd *commands.Command, args []string) { version.ShowShortVersionBanner() },
|
||||||
|
Run: GenerateCode,
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
CmdGenerate.Flag.Var(&generate.Tables, "Tables", "List of table names separated by a comma.")
|
||||||
|
CmdGenerate.Flag.Var(&generate.SQLDriver, "SQLDriver", "Database SQLDriver. Either mysql, postgres or sqlite.")
|
||||||
|
CmdGenerate.Flag.Var(&generate.SQLConn, "SQLConn", "Connection string used by the SQLDriver to connect to a database instance.")
|
||||||
|
CmdGenerate.Flag.Var(&generate.Level, "Level", "Either 1, 2 or 3. i.e. 1=models; 2=models and controllers; 3=models, controllers and routers.")
|
||||||
|
CmdGenerate.Flag.Var(&generate.Fields, "Fields", "List of table Fields.")
|
||||||
|
commands.AvailableCommands = append(commands.AvailableCommands, CmdGenerate)
|
||||||
|
}
|
||||||
|
|
||||||
|
func GenerateCode(cmd *commands.Command, args []string) int {
|
||||||
|
currpath, _ := os.Getwd()
|
||||||
|
if len(args) < 1 {
|
||||||
|
beeLogger.Log.Fatal("Command is missing")
|
||||||
|
}
|
||||||
|
|
||||||
|
gps := utils.GetGOPATHs()
|
||||||
|
if len(gps) == 0 {
|
||||||
|
beeLogger.Log.Fatal("GOPATH environment variable is not set or empty")
|
||||||
|
}
|
||||||
|
|
||||||
|
gopath := gps[0]
|
||||||
|
|
||||||
|
beeLogger.Log.Debugf("GOPATH: %s", utils.FILE(), utils.LINE(), gopath)
|
||||||
|
|
||||||
|
gcmd := args[0]
|
||||||
|
switch gcmd {
|
||||||
|
case "scaffold":
|
||||||
|
scaffold(cmd, args, currpath)
|
||||||
|
case "docs":
|
||||||
|
swaggergen.GenerateDocs(currpath)
|
||||||
|
case "appcode":
|
||||||
|
appCode(cmd, args, currpath)
|
||||||
|
case "migration":
|
||||||
|
migration(cmd, args, currpath)
|
||||||
|
case "controller":
|
||||||
|
controller(args, currpath)
|
||||||
|
case "model":
|
||||||
|
model(cmd, args, currpath)
|
||||||
|
case "view":
|
||||||
|
view(args, currpath)
|
||||||
|
default:
|
||||||
|
beeLogger.Log.Fatal("Command is missing")
|
||||||
|
}
|
||||||
|
beeLogger.Log.Successf("%s successfully generated!", strings.Title(gcmd))
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func scaffold(cmd *commands.Command, args []string, currpath string) {
|
||||||
|
if len(args) < 2 {
|
||||||
|
beeLogger.Log.Fatal("Wrong number of arguments. Run: bee help generate")
|
||||||
|
}
|
||||||
|
|
||||||
|
cmd.Flag.Parse(args[2:])
|
||||||
|
if generate.SQLDriver == "" {
|
||||||
|
generate.SQLDriver = utils.DocValue(config.Conf.Database.Driver)
|
||||||
|
if generate.SQLDriver == "" {
|
||||||
|
generate.SQLDriver = "mysql"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if generate.SQLConn == "" {
|
||||||
|
generate.SQLConn = utils.DocValue(config.Conf.Database.Conn)
|
||||||
|
if generate.SQLConn == "" {
|
||||||
|
generate.SQLConn = "root:@tcp(127.0.0.1:3306)/test"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if generate.Fields == "" {
|
||||||
|
beeLogger.Log.Hint("Fields option should not be empty, i.e. -Fields=\"title:string,body:text\"")
|
||||||
|
beeLogger.Log.Fatal("Wrong number of arguments. Run: bee help generate")
|
||||||
|
}
|
||||||
|
sname := args[1]
|
||||||
|
generate.GenerateScaffold(sname, generate.Fields.String(), currpath, generate.SQLDriver.String(), generate.SQLConn.String())
|
||||||
|
}
|
||||||
|
|
||||||
|
func appCode(cmd *commands.Command, args []string, currpath string) {
|
||||||
|
cmd.Flag.Parse(args[1:])
|
||||||
|
if generate.SQLDriver == "" {
|
||||||
|
generate.SQLDriver = utils.DocValue(config.Conf.Database.Driver)
|
||||||
|
if generate.SQLDriver == "" {
|
||||||
|
generate.SQLDriver = "mysql"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if generate.SQLConn == "" {
|
||||||
|
generate.SQLConn = utils.DocValue(config.Conf.Database.Conn)
|
||||||
|
if generate.SQLConn == "" {
|
||||||
|
if generate.SQLDriver == "mysql" {
|
||||||
|
generate.SQLConn = "root:@tcp(127.0.0.1:3306)/test"
|
||||||
|
} else if generate.SQLDriver == "postgres" {
|
||||||
|
generate.SQLConn = "postgres://postgres:postgres@127.0.0.1:5432/postgres"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if generate.Level == "" {
|
||||||
|
generate.Level = "3"
|
||||||
|
}
|
||||||
|
beeLogger.Log.Infof("Using '%s' as 'SQLDriver'", generate.SQLDriver)
|
||||||
|
beeLogger.Log.Infof("Using '%s' as 'SQLConn'", generate.SQLConn)
|
||||||
|
beeLogger.Log.Infof("Using '%s' as 'Tables'", generate.Tables)
|
||||||
|
beeLogger.Log.Infof("Using '%s' as 'Level'", generate.Level)
|
||||||
|
generate.GenerateAppcode(generate.SQLDriver.String(), generate.SQLConn.String(), generate.Level.String(), generate.Tables.String(), currpath)
|
||||||
|
}
|
||||||
|
func migration(cmd *commands.Command, args []string, currpath string) {
|
||||||
|
if len(args) < 2 {
|
||||||
|
beeLogger.Log.Fatal("Wrong number of arguments. Run: bee help generate")
|
||||||
|
}
|
||||||
|
cmd.Flag.Parse(args[2:])
|
||||||
|
mname := args[1]
|
||||||
|
|
||||||
|
beeLogger.Log.Infof("Using '%s' as migration name", mname)
|
||||||
|
|
||||||
|
upsql := ""
|
||||||
|
downsql := ""
|
||||||
|
if generate.Fields != "" {
|
||||||
|
dbMigrator := generate.NewDBDriver()
|
||||||
|
upsql = dbMigrator.GenerateCreateUp(mname)
|
||||||
|
downsql = dbMigrator.GenerateCreateDown(mname)
|
||||||
|
}
|
||||||
|
generate.GenerateMigration(mname, upsql, downsql, currpath)
|
||||||
|
}
|
||||||
|
|
||||||
|
func controller(args []string, currpath string) {
|
||||||
|
if len(args) == 2 {
|
||||||
|
cname := args[1]
|
||||||
|
generate.GenerateController(cname, currpath)
|
||||||
|
} else {
|
||||||
|
beeLogger.Log.Fatal("Wrong number of arguments. Run: bee help generate")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func model(cmd *commands.Command, args []string, currpath string) {
|
||||||
|
if len(args) < 2 {
|
||||||
|
beeLogger.Log.Fatal("Wrong number of arguments. Run: bee help generate")
|
||||||
|
}
|
||||||
|
cmd.Flag.Parse(args[2:])
|
||||||
|
if generate.Fields == "" {
|
||||||
|
beeLogger.Log.Hint("Fields option should not be empty, i.e. -Fields=\"title:string,body:text\"")
|
||||||
|
beeLogger.Log.Fatal("Wrong number of arguments. Run: bee help generate")
|
||||||
|
}
|
||||||
|
sname := args[1]
|
||||||
|
generate.GenerateModel(sname, generate.Fields.String(), currpath)
|
||||||
|
}
|
||||||
|
|
||||||
|
func view(args []string, currpath string) {
|
||||||
|
if len(args) == 2 {
|
||||||
|
cname := args[1]
|
||||||
|
generate.GenerateView(cname, currpath)
|
||||||
|
} else {
|
||||||
|
beeLogger.Log.Fatal("Wrong number of arguments. Run: bee help generate")
|
||||||
|
}
|
||||||
|
}
|
119
cmd/commands/hprose/hprose.go
Normal file
119
cmd/commands/hprose/hprose.go
Normal file
@ -0,0 +1,119 @@
|
|||||||
|
package hprose
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"fmt"
|
||||||
|
"path"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
"github.com/beego/bee/cmd/commands/api"
|
||||||
|
"github.com/beego/bee/cmd/commands/version"
|
||||||
|
"github.com/beego/bee/generate"
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
|
)
|
||||||
|
|
||||||
|
var CmdHproseapp = &commands.Command{
|
||||||
|
// CustomFlags: true,
|
||||||
|
UsageLine: "hprose [appname]",
|
||||||
|
Short: "Creates an RPC application based on Hprose and Beego frameworks",
|
||||||
|
Long: `
|
||||||
|
The command 'hprose' creates an RPC application based on both Beego and Hprose (http://hprose.com/).
|
||||||
|
|
||||||
|
{{"To scaffold out your application, use:"|bold}}
|
||||||
|
|
||||||
|
$ bee hprose [appname] [-tables=""] [-driver=mysql] [-conn=root:@tcp(127.0.0.1:3306)/test]
|
||||||
|
|
||||||
|
If 'conn' is empty, the command will generate a sample application. Otherwise the command
|
||||||
|
will connect to your database and generate models based on the existing tables.
|
||||||
|
|
||||||
|
The command 'hprose' creates a folder named [appname] with the following structure:
|
||||||
|
|
||||||
|
├── main.go
|
||||||
|
├── {{"conf"|foldername}}
|
||||||
|
│ └── app.conf
|
||||||
|
└── {{"models"|foldername}}
|
||||||
|
└── object.go
|
||||||
|
└── user.go
|
||||||
|
`,
|
||||||
|
PreRun: func(cmd *commands.Command, args []string) { version.ShowShortVersionBanner() },
|
||||||
|
Run: createhprose,
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
CmdHproseapp.Flag.Var(&generate.Tables, "tables", "List of table names separated by a comma.")
|
||||||
|
CmdHproseapp.Flag.Var(&generate.SQLDriver, "driver", "Database driver. Either mysql, postgres or sqlite.")
|
||||||
|
CmdHproseapp.Flag.Var(&generate.SQLConn, "conn", "Connection string used by the driver to connect to a database instance.")
|
||||||
|
commands.AvailableCommands = append(commands.AvailableCommands, CmdHproseapp)
|
||||||
|
}
|
||||||
|
|
||||||
|
func createhprose(cmd *commands.Command, args []string) int {
|
||||||
|
output := cmd.Out()
|
||||||
|
|
||||||
|
if len(args) != 1 {
|
||||||
|
beeLogger.Log.Fatal("Argument [appname] is missing")
|
||||||
|
}
|
||||||
|
|
||||||
|
curpath, _ := os.Getwd()
|
||||||
|
if len(args) > 1 {
|
||||||
|
cmd.Flag.Parse(args[1:])
|
||||||
|
}
|
||||||
|
apppath, packpath, err := utils.CheckEnv(args[0])
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Fatalf("%s", err)
|
||||||
|
}
|
||||||
|
if generate.SQLDriver == "" {
|
||||||
|
generate.SQLDriver = "mysql"
|
||||||
|
}
|
||||||
|
beeLogger.Log.Info("Creating Hprose application...")
|
||||||
|
|
||||||
|
os.MkdirAll(apppath, 0755)
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", apppath, "\x1b[0m")
|
||||||
|
os.Mkdir(path.Join(apppath, "conf"), 0755)
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "conf"), "\x1b[0m")
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "conf", "app.conf"), "\x1b[0m")
|
||||||
|
utils.WriteToFile(path.Join(apppath, "conf", "app.conf"),
|
||||||
|
strings.Replace(generate.Hproseconf, "{{.Appname}}", args[0], -1))
|
||||||
|
|
||||||
|
if generate.SQLConn != "" {
|
||||||
|
beeLogger.Log.Infof("Using '%s' as 'driver'", generate.SQLDriver)
|
||||||
|
beeLogger.Log.Infof("Using '%s' as 'conn'", generate.SQLConn)
|
||||||
|
beeLogger.Log.Infof("Using '%s' as 'tables'", generate.Tables)
|
||||||
|
generate.GenerateHproseAppcode(string(generate.SQLDriver), string(generate.SQLConn), "1", string(generate.Tables), path.Join(curpath, args[0]))
|
||||||
|
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "main.go"), "\x1b[0m")
|
||||||
|
maingoContent := strings.Replace(generate.HproseMainconngo, "{{.Appname}}", packpath, -1)
|
||||||
|
maingoContent = strings.Replace(maingoContent, "{{.DriverName}}", string(generate.SQLDriver), -1)
|
||||||
|
maingoContent = strings.Replace(maingoContent, "{{HproseFunctionList}}", strings.Join(generate.HproseAddFunctions, ""), -1)
|
||||||
|
if generate.SQLDriver == "mysql" {
|
||||||
|
maingoContent = strings.Replace(maingoContent, "{{.DriverPkg}}", `_ "github.com/go-sql-driver/mysql"`, -1)
|
||||||
|
} else if generate.SQLDriver == "postgres" {
|
||||||
|
maingoContent = strings.Replace(maingoContent, "{{.DriverPkg}}", `_ "github.com/lib/pq"`, -1)
|
||||||
|
}
|
||||||
|
utils.WriteToFile(path.Join(apppath, "main.go"),
|
||||||
|
strings.Replace(
|
||||||
|
maingoContent,
|
||||||
|
"{{.conn}}",
|
||||||
|
generate.SQLConn.String(),
|
||||||
|
-1,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
os.Mkdir(path.Join(apppath, "models"), 0755)
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "models"), "\x1b[0m")
|
||||||
|
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "models", "object.go"), "\x1b[0m")
|
||||||
|
utils.WriteToFile(path.Join(apppath, "models", "object.go"), apiapp.APIModels)
|
||||||
|
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "models", "user.go"), "\x1b[0m")
|
||||||
|
utils.WriteToFile(path.Join(apppath, "models", "user.go"), apiapp.APIModels2)
|
||||||
|
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "main.go"), "\x1b[0m")
|
||||||
|
utils.WriteToFile(path.Join(apppath, "main.go"),
|
||||||
|
strings.Replace(generate.HproseMaingo, "{{.Appname}}", packpath, -1))
|
||||||
|
}
|
||||||
|
beeLogger.Log.Success("New Hprose application successfully created!")
|
||||||
|
return 0
|
||||||
|
}
|
@ -11,8 +11,7 @@
|
|||||||
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
package migrate
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"database/sql"
|
"database/sql"
|
||||||
@ -23,9 +22,16 @@ import (
|
|||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
"github.com/beego/bee/cmd/commands/version"
|
||||||
|
"github.com/beego/bee/config"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
|
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
var cmdMigrate = &Command{
|
var CmdMigrate = &commands.Command{
|
||||||
UsageLine: "migrate [Command]",
|
UsageLine: "migrate [Command]",
|
||||||
Short: "Runs database migrations",
|
Short: "Runs database migrations",
|
||||||
Long: `The command 'migrate' allows you to run database migrations to keep it up-to-date.
|
Long: `The command 'migrate' allows you to run database migrations to keep it up-to-date.
|
||||||
@ -46,100 +52,75 @@ var cmdMigrate = &Command{
|
|||||||
|
|
||||||
$ bee migrate refresh [-driver=mysql] [-conn="root:@tcp(127.0.0.1:3306)/test"]
|
$ bee migrate refresh [-driver=mysql] [-conn="root:@tcp(127.0.0.1:3306)/test"]
|
||||||
`,
|
`,
|
||||||
PreRun: func(cmd *Command, args []string) { ShowShortVersionBanner() },
|
PreRun: func(cmd *commands.Command, args []string) { version.ShowShortVersionBanner() },
|
||||||
Run: runMigration,
|
Run: RunMigration,
|
||||||
}
|
}
|
||||||
|
|
||||||
var mDriver docValue
|
var mDriver utils.DocValue
|
||||||
var mConn docValue
|
var mConn utils.DocValue
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
cmdMigrate.Flag.Var(&mDriver, "driver", "Database driver. Either mysql, postgres or sqlite.")
|
CmdMigrate.Flag.Var(&mDriver, "driver", "Database driver. Either mysql, postgres or sqlite.")
|
||||||
cmdMigrate.Flag.Var(&mConn, "conn", "Connection string used by the driver to connect to a database instance.")
|
CmdMigrate.Flag.Var(&mConn, "conn", "Connection string used by the driver to connect to a database instance.")
|
||||||
|
commands.AvailableCommands = append(commands.AvailableCommands, CmdMigrate)
|
||||||
}
|
}
|
||||||
|
|
||||||
// runMigration is the entry point for starting a migration
|
// runMigration is the entry point for starting a migration
|
||||||
func runMigration(cmd *Command, args []string) int {
|
func RunMigration(cmd *commands.Command, args []string) int {
|
||||||
currpath, _ := os.Getwd()
|
currpath, _ := os.Getwd()
|
||||||
|
|
||||||
gps := GetGOPATHs()
|
gps := utils.GetGOPATHs()
|
||||||
if len(gps) == 0 {
|
if len(gps) == 0 {
|
||||||
logger.Fatal("GOPATH environment variable is not set or empty")
|
beeLogger.Log.Fatal("GOPATH environment variable is not set or empty")
|
||||||
}
|
}
|
||||||
|
|
||||||
gopath := gps[0]
|
gopath := gps[0]
|
||||||
|
|
||||||
logger.Debugf("GOPATH: %s", __FILE__(), __LINE__(), gopath)
|
beeLogger.Log.Debugf("GOPATH: %s", utils.FILE(), utils.LINE(), gopath)
|
||||||
|
|
||||||
// Load the configuration
|
|
||||||
err := loadConfig()
|
|
||||||
if err != nil {
|
|
||||||
logger.Errorf("Failed to load configuration: %s", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Getting command line arguments
|
// Getting command line arguments
|
||||||
if len(args) != 0 {
|
if len(args) != 0 {
|
||||||
cmd.Flag.Parse(args[1:])
|
cmd.Flag.Parse(args[1:])
|
||||||
}
|
}
|
||||||
if mDriver == "" {
|
if mDriver == "" {
|
||||||
mDriver = docValue(conf.Database.Driver)
|
mDriver = utils.DocValue(config.Conf.Database.Driver)
|
||||||
if mDriver == "" {
|
if mDriver == "" {
|
||||||
mDriver = "mysql"
|
mDriver = "mysql"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if mConn == "" {
|
if mConn == "" {
|
||||||
mConn = docValue(conf.Database.Conn)
|
mConn = utils.DocValue(config.Conf.Database.Conn)
|
||||||
if mConn == "" {
|
if mConn == "" {
|
||||||
mConn = "root:@tcp(127.0.0.1:3306)/test"
|
mConn = "root:@tcp(127.0.0.1:3306)/test"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
logger.Infof("Using '%s' as 'driver'", mDriver)
|
beeLogger.Log.Infof("Using '%s' as 'driver'", mDriver)
|
||||||
logger.Infof("Using '%s' as 'conn'", mConn)
|
beeLogger.Log.Infof("Using '%s' as 'conn'", mConn)
|
||||||
driverStr, connStr := string(mDriver), string(mConn)
|
driverStr, connStr := string(mDriver), string(mConn)
|
||||||
if len(args) == 0 {
|
if len(args) == 0 {
|
||||||
// run all outstanding migrations
|
// run all outstanding migrations
|
||||||
logger.Info("Running all outstanding migrations")
|
beeLogger.Log.Info("Running all outstanding migrations")
|
||||||
migrateUpdate(currpath, driverStr, connStr)
|
MigrateUpdate(currpath, driverStr, connStr)
|
||||||
} else {
|
} else {
|
||||||
mcmd := args[0]
|
mcmd := args[0]
|
||||||
switch mcmd {
|
switch mcmd {
|
||||||
case "rollback":
|
case "rollback":
|
||||||
logger.Info("Rolling back the last migration operation")
|
beeLogger.Log.Info("Rolling back the last migration operation")
|
||||||
migrateRollback(currpath, driverStr, connStr)
|
MigrateRollback(currpath, driverStr, connStr)
|
||||||
case "reset":
|
case "reset":
|
||||||
logger.Info("Reseting all migrations")
|
beeLogger.Log.Info("Reseting all migrations")
|
||||||
migrateReset(currpath, driverStr, connStr)
|
MigrateReset(currpath, driverStr, connStr)
|
||||||
case "refresh":
|
case "refresh":
|
||||||
logger.Info("Refreshing all migrations")
|
beeLogger.Log.Info("Refreshing all migrations")
|
||||||
migrateRefresh(currpath, driverStr, connStr)
|
MigrateRefresh(currpath, driverStr, connStr)
|
||||||
default:
|
default:
|
||||||
logger.Fatal("Command is missing")
|
beeLogger.Log.Fatal("Command is missing")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
logger.Success("Migration successful!")
|
beeLogger.Log.Success("Migration successful!")
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
// migrateUpdate does the schema update
|
|
||||||
func migrateUpdate(currpath, driver, connStr string) {
|
|
||||||
migrate("upgrade", currpath, driver, connStr)
|
|
||||||
}
|
|
||||||
|
|
||||||
// migrateRollback rolls back the latest migration
|
|
||||||
func migrateRollback(currpath, driver, connStr string) {
|
|
||||||
migrate("rollback", currpath, driver, connStr)
|
|
||||||
}
|
|
||||||
|
|
||||||
// migrateReset rolls back all migrations
|
|
||||||
func migrateReset(currpath, driver, connStr string) {
|
|
||||||
migrate("reset", currpath, driver, connStr)
|
|
||||||
}
|
|
||||||
|
|
||||||
// migrationRefresh rolls back all migrations and start over again
|
|
||||||
func migrateRefresh(currpath, driver, connStr string) {
|
|
||||||
migrate("refresh", currpath, driver, connStr)
|
|
||||||
}
|
|
||||||
|
|
||||||
// migrate generates source code, build it, and invoke the binary who does the actual migration
|
// migrate generates source code, build it, and invoke the binary who does the actual migration
|
||||||
func migrate(goal, currpath, driver, connStr string) {
|
func migrate(goal, currpath, driver, connStr string) {
|
||||||
dir := path.Join(currpath, "database", "migrations")
|
dir := path.Join(currpath, "database", "migrations")
|
||||||
@ -153,7 +134,7 @@ func migrate(goal, currpath, driver, connStr string) {
|
|||||||
// Connect to database
|
// Connect to database
|
||||||
db, err := sql.Open(driver, connStr)
|
db, err := sql.Open(driver, connStr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Could not connect to database using '%s': %s", connStr, err)
|
beeLogger.Log.Fatalf("Could not connect to database using '%s': %s", connStr, err)
|
||||||
}
|
}
|
||||||
defer db.Close()
|
defer db.Close()
|
||||||
|
|
||||||
@ -171,44 +152,44 @@ func migrate(goal, currpath, driver, connStr string) {
|
|||||||
func checkForSchemaUpdateTable(db *sql.DB, driver string) {
|
func checkForSchemaUpdateTable(db *sql.DB, driver string) {
|
||||||
showTableSQL := showMigrationsTableSQL(driver)
|
showTableSQL := showMigrationsTableSQL(driver)
|
||||||
if rows, err := db.Query(showTableSQL); err != nil {
|
if rows, err := db.Query(showTableSQL); err != nil {
|
||||||
logger.Fatalf("Could not show migrations table: %s", err)
|
beeLogger.Log.Fatalf("Could not show migrations table: %s", err)
|
||||||
} else if !rows.Next() {
|
} else if !rows.Next() {
|
||||||
// No migrations table, create new ones
|
// No migrations table, create new ones
|
||||||
createTableSQL := createMigrationsTableSQL(driver)
|
createTableSQL := createMigrationsTableSQL(driver)
|
||||||
|
|
||||||
logger.Infof("Creating 'migrations' table...")
|
beeLogger.Log.Infof("Creating 'migrations' table...")
|
||||||
|
|
||||||
if _, err := db.Query(createTableSQL); err != nil {
|
if _, err := db.Query(createTableSQL); err != nil {
|
||||||
logger.Fatalf("Could not create migrations table: %s", err)
|
beeLogger.Log.Fatalf("Could not create migrations table: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Checking that migrations table schema are expected
|
// Checking that migrations table schema are expected
|
||||||
selectTableSQL := selectMigrationsTableSQL(driver)
|
selectTableSQL := selectMigrationsTableSQL(driver)
|
||||||
if rows, err := db.Query(selectTableSQL); err != nil {
|
if rows, err := db.Query(selectTableSQL); err != nil {
|
||||||
logger.Fatalf("Could not show columns of migrations table: %s", err)
|
beeLogger.Log.Fatalf("Could not show columns of migrations table: %s", err)
|
||||||
} else {
|
} else {
|
||||||
for rows.Next() {
|
for rows.Next() {
|
||||||
var fieldBytes, typeBytes, nullBytes, keyBytes, defaultBytes, extraBytes []byte
|
var fieldBytes, typeBytes, nullBytes, keyBytes, defaultBytes, extraBytes []byte
|
||||||
if err := rows.Scan(&fieldBytes, &typeBytes, &nullBytes, &keyBytes, &defaultBytes, &extraBytes); err != nil {
|
if err := rows.Scan(&fieldBytes, &typeBytes, &nullBytes, &keyBytes, &defaultBytes, &extraBytes); err != nil {
|
||||||
logger.Fatalf("Could not read column information: %s", err)
|
beeLogger.Log.Fatalf("Could not read column information: %s", err)
|
||||||
}
|
}
|
||||||
fieldStr, typeStr, nullStr, keyStr, defaultStr, extraStr :=
|
fieldStr, typeStr, nullStr, keyStr, defaultStr, extraStr :=
|
||||||
string(fieldBytes), string(typeBytes), string(nullBytes), string(keyBytes), string(defaultBytes), string(extraBytes)
|
string(fieldBytes), string(typeBytes), string(nullBytes), string(keyBytes), string(defaultBytes), string(extraBytes)
|
||||||
if fieldStr == "id_migration" {
|
if fieldStr == "id_migration" {
|
||||||
if keyStr != "PRI" || extraStr != "auto_increment" {
|
if keyStr != "PRI" || extraStr != "auto_increment" {
|
||||||
logger.Hint("Expecting KEY: PRI, EXTRA: auto_increment")
|
beeLogger.Log.Hint("Expecting KEY: PRI, EXTRA: auto_increment")
|
||||||
logger.Fatalf("Column migration.id_migration type mismatch: KEY: %s, EXTRA: %s", keyStr, extraStr)
|
beeLogger.Log.Fatalf("Column migration.id_migration type mismatch: KEY: %s, EXTRA: %s", keyStr, extraStr)
|
||||||
}
|
}
|
||||||
} else if fieldStr == "name" {
|
} else if fieldStr == "name" {
|
||||||
if !strings.HasPrefix(typeStr, "varchar") || nullStr != "YES" {
|
if !strings.HasPrefix(typeStr, "varchar") || nullStr != "YES" {
|
||||||
logger.Hint("Expecting TYPE: varchar, NULL: YES")
|
beeLogger.Log.Hint("Expecting TYPE: varchar, NULL: YES")
|
||||||
logger.Fatalf("Column migration.name type mismatch: TYPE: %s, NULL: %s", typeStr, nullStr)
|
beeLogger.Log.Fatalf("Column migration.name type mismatch: TYPE: %s, NULL: %s", typeStr, nullStr)
|
||||||
}
|
}
|
||||||
} else if fieldStr == "created_at" {
|
} else if fieldStr == "created_at" {
|
||||||
if typeStr != "timestamp" || defaultStr != "CURRENT_TIMESTAMP" {
|
if typeStr != "timestamp" || defaultStr != "CURRENT_TIMESTAMP" {
|
||||||
logger.Hint("Expecting TYPE: timestamp, DEFAULT: CURRENT_TIMESTAMP")
|
beeLogger.Log.Hint("Expecting TYPE: timestamp, DEFAULT: CURRENT_TIMESTAMP")
|
||||||
logger.Fatalf("Column migration.timestamp type mismatch: TYPE: %s, DEFAULT: %s", typeStr, defaultStr)
|
beeLogger.Log.Fatalf("Column migration.timestamp type mismatch: TYPE: %s, DEFAULT: %s", typeStr, defaultStr)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -252,22 +233,22 @@ func selectMigrationsTableSQL(driver string) string {
|
|||||||
func getLatestMigration(db *sql.DB, goal string) (file string, createdAt int64) {
|
func getLatestMigration(db *sql.DB, goal string) (file string, createdAt int64) {
|
||||||
sql := "SELECT name FROM migrations where status = 'update' ORDER BY id_migration DESC LIMIT 1"
|
sql := "SELECT name FROM migrations where status = 'update' ORDER BY id_migration DESC LIMIT 1"
|
||||||
if rows, err := db.Query(sql); err != nil {
|
if rows, err := db.Query(sql); err != nil {
|
||||||
logger.Fatalf("Could not retrieve migrations: %s", err)
|
beeLogger.Log.Fatalf("Could not retrieve migrations: %s", err)
|
||||||
} else {
|
} else {
|
||||||
if rows.Next() {
|
if rows.Next() {
|
||||||
if err := rows.Scan(&file); err != nil {
|
if err := rows.Scan(&file); err != nil {
|
||||||
logger.Fatalf("Could not read migrations in database: %s", err)
|
beeLogger.Log.Fatalf("Could not read migrations in database: %s", err)
|
||||||
}
|
}
|
||||||
createdAtStr := file[len(file)-15:]
|
createdAtStr := file[len(file)-15:]
|
||||||
if t, err := time.Parse("20060102_150405", createdAtStr); err != nil {
|
if t, err := time.Parse("20060102_150405", createdAtStr); err != nil {
|
||||||
logger.Fatalf("Could not parse time: %s", err)
|
beeLogger.Log.Fatalf("Could not parse time: %s", err)
|
||||||
} else {
|
} else {
|
||||||
createdAt = t.Unix()
|
createdAt = t.Unix()
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// migration table has no 'update' record, no point rolling back
|
// migration table has no 'update' record, no point rolling back
|
||||||
if goal == "rollback" {
|
if goal == "rollback" {
|
||||||
logger.Fatal("There is nothing to rollback")
|
beeLogger.Log.Fatal("There is nothing to rollback")
|
||||||
}
|
}
|
||||||
file, createdAt = "", 0
|
file, createdAt = "", 0
|
||||||
}
|
}
|
||||||
@ -279,7 +260,7 @@ func getLatestMigration(db *sql.DB, goal string) (file string, createdAt int64)
|
|||||||
func writeMigrationSourceFile(dir, source, driver, connStr string, latestTime int64, latestName string, task string) {
|
func writeMigrationSourceFile(dir, source, driver, connStr string, latestTime int64, latestName string, task string) {
|
||||||
changeDir(dir)
|
changeDir(dir)
|
||||||
if f, err := os.OpenFile(source, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err != nil {
|
if f, err := os.OpenFile(source, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err != nil {
|
||||||
logger.Fatalf("Could not create file: %s", err)
|
beeLogger.Log.Fatalf("Could not create file: %s", err)
|
||||||
} else {
|
} else {
|
||||||
content := strings.Replace(MigrationMainTPL, "{{DBDriver}}", driver, -1)
|
content := strings.Replace(MigrationMainTPL, "{{DBDriver}}", driver, -1)
|
||||||
content = strings.Replace(content, "{{ConnStr}}", connStr, -1)
|
content = strings.Replace(content, "{{ConnStr}}", connStr, -1)
|
||||||
@ -287,9 +268,9 @@ func writeMigrationSourceFile(dir, source, driver, connStr string, latestTime in
|
|||||||
content = strings.Replace(content, "{{LatestName}}", latestName, -1)
|
content = strings.Replace(content, "{{LatestName}}", latestName, -1)
|
||||||
content = strings.Replace(content, "{{Task}}", task, -1)
|
content = strings.Replace(content, "{{Task}}", task, -1)
|
||||||
if _, err := f.WriteString(content); err != nil {
|
if _, err := f.WriteString(content); err != nil {
|
||||||
logger.Fatalf("Could not write to file: %s", err)
|
beeLogger.Log.Fatalf("Could not write to file: %s", err)
|
||||||
}
|
}
|
||||||
CloseFile(f)
|
utils.CloseFile(f)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -298,7 +279,7 @@ func buildMigrationBinary(dir, binary string) {
|
|||||||
changeDir(dir)
|
changeDir(dir)
|
||||||
cmd := exec.Command("go", "build", "-o", binary)
|
cmd := exec.Command("go", "build", "-o", binary)
|
||||||
if out, err := cmd.CombinedOutput(); err != nil {
|
if out, err := cmd.CombinedOutput(); err != nil {
|
||||||
logger.Errorf("Could not build migration binary: %s", err)
|
beeLogger.Log.Errorf("Could not build migration binary: %s", err)
|
||||||
formatShellErrOutput(string(out))
|
formatShellErrOutput(string(out))
|
||||||
removeTempFile(dir, binary)
|
removeTempFile(dir, binary)
|
||||||
removeTempFile(dir, binary+".go")
|
removeTempFile(dir, binary+".go")
|
||||||
@ -312,7 +293,7 @@ func runMigrationBinary(dir, binary string) {
|
|||||||
cmd := exec.Command("./" + binary)
|
cmd := exec.Command("./" + binary)
|
||||||
if out, err := cmd.CombinedOutput(); err != nil {
|
if out, err := cmd.CombinedOutput(); err != nil {
|
||||||
formatShellOutput(string(out))
|
formatShellOutput(string(out))
|
||||||
logger.Errorf("Could not run migration binary: %s", err)
|
beeLogger.Log.Errorf("Could not run migration binary: %s", err)
|
||||||
removeTempFile(dir, binary)
|
removeTempFile(dir, binary)
|
||||||
removeTempFile(dir, binary+".go")
|
removeTempFile(dir, binary+".go")
|
||||||
os.Exit(2)
|
os.Exit(2)
|
||||||
@ -325,7 +306,7 @@ func runMigrationBinary(dir, binary string) {
|
|||||||
// It exits the system when encouter an error
|
// It exits the system when encouter an error
|
||||||
func changeDir(dir string) {
|
func changeDir(dir string) {
|
||||||
if err := os.Chdir(dir); err != nil {
|
if err := os.Chdir(dir); err != nil {
|
||||||
logger.Fatalf("Could not find migration directory: %s", err)
|
beeLogger.Log.Fatalf("Could not find migration directory: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -333,7 +314,7 @@ func changeDir(dir string) {
|
|||||||
func removeTempFile(dir, file string) {
|
func removeTempFile(dir, file string) {
|
||||||
changeDir(dir)
|
changeDir(dir)
|
||||||
if err := os.Remove(file); err != nil {
|
if err := os.Remove(file); err != nil {
|
||||||
logger.Warnf("Could not remove temporary file: %s", err)
|
beeLogger.Log.Warnf("Could not remove temporary file: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -341,7 +322,7 @@ func removeTempFile(dir, file string) {
|
|||||||
func formatShellErrOutput(o string) {
|
func formatShellErrOutput(o string) {
|
||||||
for _, line := range strings.Split(o, "\n") {
|
for _, line := range strings.Split(o, "\n") {
|
||||||
if line != "" {
|
if line != "" {
|
||||||
logger.Errorf("|> %s", line)
|
beeLogger.Log.Errorf("|> %s", line)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -350,7 +331,7 @@ func formatShellErrOutput(o string) {
|
|||||||
func formatShellOutput(o string) {
|
func formatShellOutput(o string) {
|
||||||
for _, line := range strings.Split(o, "\n") {
|
for _, line := range strings.Split(o, "\n") {
|
||||||
if line != "" {
|
if line != "" {
|
||||||
logger.Infof("|> %s", line)
|
beeLogger.Log.Infof("|> %s", line)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -421,3 +402,23 @@ CREATE TABLE migrations (
|
|||||||
status migrations_status
|
status migrations_status
|
||||||
)`
|
)`
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// MigrateUpdate does the schema update
|
||||||
|
func MigrateUpdate(currpath, driver, connStr string) {
|
||||||
|
migrate("upgrade", currpath, driver, connStr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// MigrateRollback rolls back the latest migration
|
||||||
|
func MigrateRollback(currpath, driver, connStr string) {
|
||||||
|
migrate("rollback", currpath, driver, connStr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// MigrateReset rolls back all migrations
|
||||||
|
func MigrateReset(currpath, driver, connStr string) {
|
||||||
|
migrate("reset", currpath, driver, connStr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// migrationRefresh rolls back all migrations and start over again
|
||||||
|
func MigrateRefresh(currpath, driver, connStr string) {
|
||||||
|
migrate("refresh", currpath, driver, connStr)
|
||||||
|
}
|
@ -12,16 +12,22 @@
|
|||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
|
||||||
package main
|
package new
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
path "path/filepath"
|
path "path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
"github.com/beego/bee/cmd/commands/version"
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/logger/colors"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
var cmdNew = &Command{
|
var CmdNew = &commands.Command{
|
||||||
UsageLine: "new [appname]",
|
UsageLine: "new [appname]",
|
||||||
Short: "Creates a Beego application",
|
Short: "Creates a Beego application",
|
||||||
Long: `
|
Long: `
|
||||||
@ -47,74 +53,8 @@ Creates a Beego application for the given app name in the current directory.
|
|||||||
└── index.tpl
|
└── index.tpl
|
||||||
|
|
||||||
`,
|
`,
|
||||||
PreRun: func(cmd *Command, args []string) { ShowShortVersionBanner() },
|
PreRun: func(cmd *commands.Command, args []string) { version.ShowShortVersionBanner() },
|
||||||
Run: createApp,
|
Run: CreateApp,
|
||||||
}
|
|
||||||
|
|
||||||
func createApp(cmd *Command, args []string) int {
|
|
||||||
output := cmd.Out()
|
|
||||||
if len(args) != 1 {
|
|
||||||
logger.Fatal("Argument [appname] is missing")
|
|
||||||
}
|
|
||||||
|
|
||||||
apppath, packpath, err := checkEnv(args[0])
|
|
||||||
if err != nil {
|
|
||||||
logger.Fatalf("%s", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if isExist(apppath) {
|
|
||||||
logger.Errorf(bold("Application '%s' already exists"), apppath)
|
|
||||||
logger.Warn(bold("Do you want to overwrite it? [Yes|No] "))
|
|
||||||
if !askForConfirmation() {
|
|
||||||
os.Exit(2)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
logger.Info("Creating application...")
|
|
||||||
|
|
||||||
os.MkdirAll(apppath, 0755)
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", apppath+string(path.Separator), "\x1b[0m")
|
|
||||||
os.Mkdir(path.Join(apppath, "conf"), 0755)
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "conf")+string(path.Separator), "\x1b[0m")
|
|
||||||
os.Mkdir(path.Join(apppath, "controllers"), 0755)
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "controllers")+string(path.Separator), "\x1b[0m")
|
|
||||||
os.Mkdir(path.Join(apppath, "models"), 0755)
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "models")+string(path.Separator), "\x1b[0m")
|
|
||||||
os.Mkdir(path.Join(apppath, "routers"), 0755)
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "routers")+string(path.Separator), "\x1b[0m")
|
|
||||||
os.Mkdir(path.Join(apppath, "tests"), 0755)
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "tests")+string(path.Separator), "\x1b[0m")
|
|
||||||
os.Mkdir(path.Join(apppath, "static"), 0755)
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "static")+string(path.Separator), "\x1b[0m")
|
|
||||||
os.Mkdir(path.Join(apppath, "static", "js"), 0755)
|
|
||||||
WriteToFile(path.Join(apppath, "static", "js", "reload.min.js"), reloadJsClient)
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "static", "js")+string(path.Separator), "\x1b[0m")
|
|
||||||
os.Mkdir(path.Join(apppath, "static", "css"), 0755)
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "static", "css")+string(path.Separator), "\x1b[0m")
|
|
||||||
os.Mkdir(path.Join(apppath, "static", "img"), 0755)
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "static", "img")+string(path.Separator), "\x1b[0m")
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "views")+string(path.Separator), "\x1b[0m")
|
|
||||||
os.Mkdir(path.Join(apppath, "views"), 0755)
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "conf", "app.conf"), "\x1b[0m")
|
|
||||||
WriteToFile(path.Join(apppath, "conf", "app.conf"), strings.Replace(appconf, "{{.Appname}}", path.Base(args[0]), -1))
|
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "controllers", "default.go"), "\x1b[0m")
|
|
||||||
WriteToFile(path.Join(apppath, "controllers", "default.go"), controllers)
|
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "views", "index.tpl"), "\x1b[0m")
|
|
||||||
WriteToFile(path.Join(apppath, "views", "index.tpl"), indextpl)
|
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "routers", "router.go"), "\x1b[0m")
|
|
||||||
WriteToFile(path.Join(apppath, "routers", "router.go"), strings.Replace(router, "{{.Appname}}", packpath, -1))
|
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "tests", "default_test.go"), "\x1b[0m")
|
|
||||||
WriteToFile(path.Join(apppath, "tests", "default_test.go"), strings.Replace(test, "{{.Appname}}", packpath, -1))
|
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "main.go"), "\x1b[0m")
|
|
||||||
WriteToFile(path.Join(apppath, "main.go"), strings.Replace(maingo, "{{.Appname}}", packpath, -1))
|
|
||||||
|
|
||||||
logger.Success("New application successfully created!")
|
|
||||||
return 0
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var appconf = `appname = {{.Appname}}
|
var appconf = `appname = {{.Appname}}
|
||||||
@ -303,3 +243,73 @@ var indextpl = `<!DOCTYPE html>
|
|||||||
|
|
||||||
var reloadJsClient = `function b(a){var c=new WebSocket(a);c.onclose=function(){setTimeout(function(){b(a)},2E3)};c.onmessage=function(){location.reload()}}try{if(window.WebSocket)try{b("ws://localhost:12450/reload")}catch(a){console.error(a)}else console.log("Your browser does not support WebSockets.")}catch(a){console.error("Exception during connecting to Reload:",a)};
|
var reloadJsClient = `function b(a){var c=new WebSocket(a);c.onclose=function(){setTimeout(function(){b(a)},2E3)};c.onmessage=function(){location.reload()}}try{if(window.WebSocket)try{b("ws://localhost:12450/reload")}catch(a){console.error(a)}else console.log("Your browser does not support WebSockets.")}catch(a){console.error("Exception during connecting to Reload:",a)};
|
||||||
`
|
`
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
commands.AvailableCommands = append(commands.AvailableCommands, CmdNew)
|
||||||
|
}
|
||||||
|
|
||||||
|
func CreateApp(cmd *commands.Command, args []string) int {
|
||||||
|
output := cmd.Out()
|
||||||
|
if len(args) != 1 {
|
||||||
|
beeLogger.Log.Fatal("Argument [appname] is missing")
|
||||||
|
}
|
||||||
|
|
||||||
|
apppath, packpath, err := utils.CheckEnv(args[0])
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Fatalf("%s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if utils.IsExist(apppath) {
|
||||||
|
beeLogger.Log.Errorf(colors.Bold("Application '%s' already exists"), apppath)
|
||||||
|
beeLogger.Log.Warn(colors.Bold("Do you want to overwrite it? [Yes|No] "))
|
||||||
|
if !utils.AskForConfirmation() {
|
||||||
|
os.Exit(2)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
beeLogger.Log.Info("Creating application...")
|
||||||
|
|
||||||
|
os.MkdirAll(apppath, 0755)
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", apppath+string(path.Separator), "\x1b[0m")
|
||||||
|
os.Mkdir(path.Join(apppath, "conf"), 0755)
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "conf")+string(path.Separator), "\x1b[0m")
|
||||||
|
os.Mkdir(path.Join(apppath, "controllers"), 0755)
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "controllers")+string(path.Separator), "\x1b[0m")
|
||||||
|
os.Mkdir(path.Join(apppath, "models"), 0755)
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "models")+string(path.Separator), "\x1b[0m")
|
||||||
|
os.Mkdir(path.Join(apppath, "routers"), 0755)
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "routers")+string(path.Separator), "\x1b[0m")
|
||||||
|
os.Mkdir(path.Join(apppath, "tests"), 0755)
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "tests")+string(path.Separator), "\x1b[0m")
|
||||||
|
os.Mkdir(path.Join(apppath, "static"), 0755)
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "static")+string(path.Separator), "\x1b[0m")
|
||||||
|
os.Mkdir(path.Join(apppath, "static", "js"), 0755)
|
||||||
|
utils.WriteToFile(path.Join(apppath, "static", "js", "reload.min.js"), reloadJsClient)
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "static", "js")+string(path.Separator), "\x1b[0m")
|
||||||
|
os.Mkdir(path.Join(apppath, "static", "css"), 0755)
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "static", "css")+string(path.Separator), "\x1b[0m")
|
||||||
|
os.Mkdir(path.Join(apppath, "static", "img"), 0755)
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "static", "img")+string(path.Separator), "\x1b[0m")
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "views")+string(path.Separator), "\x1b[0m")
|
||||||
|
os.Mkdir(path.Join(apppath, "views"), 0755)
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "conf", "app.conf"), "\x1b[0m")
|
||||||
|
utils.WriteToFile(path.Join(apppath, "conf", "app.conf"), strings.Replace(appconf, "{{.Appname}}", path.Base(args[0]), -1))
|
||||||
|
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "controllers", "default.go"), "\x1b[0m")
|
||||||
|
utils.WriteToFile(path.Join(apppath, "controllers", "default.go"), controllers)
|
||||||
|
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "views", "index.tpl"), "\x1b[0m")
|
||||||
|
utils.WriteToFile(path.Join(apppath, "views", "index.tpl"), indextpl)
|
||||||
|
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "routers", "router.go"), "\x1b[0m")
|
||||||
|
utils.WriteToFile(path.Join(apppath, "routers", "router.go"), strings.Replace(router, "{{.Appname}}", packpath, -1))
|
||||||
|
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "tests", "default_test.go"), "\x1b[0m")
|
||||||
|
utils.WriteToFile(path.Join(apppath, "tests", "default_test.go"), strings.Replace(test, "{{.Appname}}", packpath, -1))
|
||||||
|
|
||||||
|
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "main.go"), "\x1b[0m")
|
||||||
|
utils.WriteToFile(path.Join(apppath, "main.go"), strings.Replace(maingo, "{{.Appname}}", packpath, -1))
|
||||||
|
|
||||||
|
beeLogger.Log.Success("New application successfully created!")
|
||||||
|
return 0
|
||||||
|
}
|
@ -1,18 +1,4 @@
|
|||||||
// Copyright 2013 bee authors
|
package pack
|
||||||
//
|
|
||||||
// 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 main
|
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"archive/tar"
|
"archive/tar"
|
||||||
@ -31,9 +17,14 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
"syscall"
|
"syscall"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
"github.com/beego/bee/cmd/commands/version"
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
var cmdPack = &Command{
|
var CmdPack = &commands.Command{
|
||||||
CustomFlags: true,
|
CustomFlags: true,
|
||||||
UsageLine: "pack",
|
UsageLine: "pack",
|
||||||
Short: "Compresses a Beego application into a single file",
|
Short: "Compresses a Beego application into a single file",
|
||||||
@ -43,7 +34,7 @@ var cmdPack = &Command{
|
|||||||
{{"Example:"|bold}}
|
{{"Example:"|bold}}
|
||||||
$ bee pack -v -ba="-ldflags '-s -w'"
|
$ bee pack -v -ba="-ldflags '-s -w'"
|
||||||
`,
|
`,
|
||||||
PreRun: func(cmd *Command, args []string) { ShowShortVersionBanner() },
|
PreRun: func(cmd *commands.Command, args []string) { version.ShowShortVersionBanner() },
|
||||||
Run: packApp,
|
Run: packApp,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -52,27 +43,16 @@ var (
|
|||||||
excludeP string
|
excludeP string
|
||||||
excludeS string
|
excludeS string
|
||||||
outputP string
|
outputP string
|
||||||
excludeR ListOpts
|
excludeR utils.ListOpts
|
||||||
fsym bool
|
fsym bool
|
||||||
ssym bool
|
ssym bool
|
||||||
build bool
|
build bool
|
||||||
buildArgs string
|
buildArgs string
|
||||||
buildEnvs ListOpts
|
buildEnvs utils.ListOpts
|
||||||
verbose bool
|
verbose bool
|
||||||
format string
|
format string
|
||||||
)
|
)
|
||||||
|
|
||||||
type ListOpts []string
|
|
||||||
|
|
||||||
func (opts *ListOpts) String() string {
|
|
||||||
return fmt.Sprint(*opts)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (opts *ListOpts) Set(value string) error {
|
|
||||||
*opts = append(*opts, value)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
fs := flag.NewFlagSet("pack", flag.ContinueOnError)
|
fs := flag.NewFlagSet("pack", flag.ContinueOnError)
|
||||||
fs.StringVar(&appPath, "p", "", "Set the application path. Defaults to the current path.")
|
fs.StringVar(&appPath, "p", "", "Set the application path. Defaults to the current path.")
|
||||||
@ -87,7 +67,8 @@ func init() {
|
|||||||
fs.BoolVar(&fsym, "fs", false, "Tell the command to follow symlinks. Defaults to false.")
|
fs.BoolVar(&fsym, "fs", false, "Tell the command to follow symlinks. Defaults to false.")
|
||||||
fs.BoolVar(&ssym, "ss", false, "Tell the command to skip symlinks. Defaults to false.")
|
fs.BoolVar(&ssym, "ss", false, "Tell the command to skip symlinks. Defaults to false.")
|
||||||
fs.BoolVar(&verbose, "v", false, "Be more verbose during the operation. Defaults to false.")
|
fs.BoolVar(&verbose, "v", false, "Be more verbose during the operation. Defaults to false.")
|
||||||
cmdPack.Flag = *fs
|
CmdPack.Flag = *fs
|
||||||
|
commands.AvailableCommands = append(commands.AvailableCommands, CmdPack)
|
||||||
}
|
}
|
||||||
|
|
||||||
type walker interface {
|
type walker interface {
|
||||||
@ -115,10 +96,6 @@ type walkFileTree struct {
|
|||||||
output *io.Writer
|
output *io.Writer
|
||||||
}
|
}
|
||||||
|
|
||||||
func (wft *walkFileTree) setPrefix(prefix string) {
|
|
||||||
wft.prefix = prefix
|
|
||||||
}
|
|
||||||
|
|
||||||
func (wft *walkFileTree) isExclude(fPath string) bool {
|
func (wft *walkFileTree) isExclude(fPath string) bool {
|
||||||
if fPath == "" {
|
if fPath == "" {
|
||||||
return true
|
return true
|
||||||
@ -316,12 +293,12 @@ func (wft *tarWalk) compress(name, fpath string, fi os.FileInfo) (bool, error) {
|
|||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
|
|
||||||
if isSym == false {
|
if !isSym {
|
||||||
fr, err := os.Open(fpath)
|
fr, err := os.Open(fpath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
defer CloseFile(fr)
|
defer utils.CloseFile(fr)
|
||||||
_, err = io.Copy(tw, fr)
|
_, err = io.Copy(tw, fr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, err
|
return false, err
|
||||||
@ -352,12 +329,12 @@ func (wft *zipWalk) compress(name, fpath string, fi os.FileInfo) (bool, error) {
|
|||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
|
|
||||||
if isSym == false {
|
if !isSym {
|
||||||
fr, err := os.Open(fpath)
|
fr, err := os.Open(fpath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
defer CloseFile(fr)
|
defer utils.CloseFile(fr)
|
||||||
_, err = io.Copy(w, fr)
|
_, err = io.Copy(w, fr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, err
|
return false, err
|
||||||
@ -379,10 +356,10 @@ func (wft *zipWalk) compress(name, fpath string, fi os.FileInfo) (bool, error) {
|
|||||||
func packDirectory(output io.Writer, excludePrefix []string, excludeSuffix []string,
|
func packDirectory(output io.Writer, excludePrefix []string, excludeSuffix []string,
|
||||||
excludeRegexp []*regexp.Regexp, includePath ...string) (err error) {
|
excludeRegexp []*regexp.Regexp, includePath ...string) (err error) {
|
||||||
|
|
||||||
logger.Infof("Excluding relpath prefix: %s", strings.Join(excludePrefix, ":"))
|
beeLogger.Log.Infof("Excluding relpath prefix: %s", strings.Join(excludePrefix, ":"))
|
||||||
logger.Infof("Excluding relpath suffix: %s", strings.Join(excludeSuffix, ":"))
|
beeLogger.Log.Infof("Excluding relpath suffix: %s", strings.Join(excludeSuffix, ":"))
|
||||||
if len(excludeRegexp) > 0 {
|
if len(excludeRegexp) > 0 {
|
||||||
logger.Infof("Excluding filename regex: `%s`", strings.Join(excludeR, "`, `"))
|
beeLogger.Log.Infof("Excluding filename regex: `%s`", strings.Join(excludeR, "`, `"))
|
||||||
}
|
}
|
||||||
|
|
||||||
w, err := os.OpenFile(outputP, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0644)
|
w, err := os.OpenFile(outputP, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0644)
|
||||||
@ -437,7 +414,7 @@ func packDirectory(output io.Writer, excludePrefix []string, excludeSuffix []str
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func packApp(cmd *Command, args []string) int {
|
func packApp(cmd *commands.Command, args []string) int {
|
||||||
output := cmd.Out()
|
output := cmd.Out()
|
||||||
curPath, _ := os.Getwd()
|
curPath, _ := os.Getwd()
|
||||||
thePath := ""
|
thePath := ""
|
||||||
@ -454,19 +431,19 @@ func packApp(cmd *Command, args []string) int {
|
|||||||
}
|
}
|
||||||
cmd.Flag.Parse(nArgs)
|
cmd.Flag.Parse(nArgs)
|
||||||
|
|
||||||
if path.IsAbs(appPath) == false {
|
if !path.IsAbs(appPath) {
|
||||||
appPath = path.Join(curPath, appPath)
|
appPath = path.Join(curPath, appPath)
|
||||||
}
|
}
|
||||||
|
|
||||||
thePath, err := path.Abs(appPath)
|
thePath, err := path.Abs(appPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Wrong application path: %s", thePath)
|
beeLogger.Log.Fatalf("Wrong application path: %s", thePath)
|
||||||
}
|
}
|
||||||
if stat, err := os.Stat(thePath); os.IsNotExist(err) || stat.IsDir() == false {
|
if stat, err := os.Stat(thePath); os.IsNotExist(err) || !stat.IsDir() {
|
||||||
logger.Fatalf("Application path does not exist: %s", thePath)
|
beeLogger.Log.Fatalf("Application path does not exist: %s", thePath)
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.Infof("Packaging application on '%s'...", thePath)
|
beeLogger.Log.Infof("Packaging application on '%s'...", thePath)
|
||||||
|
|
||||||
appName := path.Base(thePath)
|
appName := path.Base(thePath)
|
||||||
|
|
||||||
@ -488,12 +465,12 @@ func packApp(cmd *Command, args []string) int {
|
|||||||
// Remove the tmpdir once bee pack is done
|
// Remove the tmpdir once bee pack is done
|
||||||
err := os.RemoveAll(tmpdir)
|
err := os.RemoveAll(tmpdir)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Error("Failed to remove the generated temp dir")
|
beeLogger.Log.Error("Failed to remove the generated temp dir")
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
if build {
|
if build {
|
||||||
logger.Info("Building application...")
|
beeLogger.Log.Info("Building application...")
|
||||||
var envs []string
|
var envs []string
|
||||||
for _, env := range buildEnvs {
|
for _, env := range buildEnvs {
|
||||||
parts := strings.SplitN(env, "=", 2)
|
parts := strings.SplitN(env, "=", 2)
|
||||||
@ -515,7 +492,7 @@ func packApp(cmd *Command, args []string) int {
|
|||||||
os.Setenv("GOOS", goos)
|
os.Setenv("GOOS", goos)
|
||||||
os.Setenv("GOARCH", goarch)
|
os.Setenv("GOARCH", goarch)
|
||||||
|
|
||||||
logger.Infof("Using: GOOS=%s GOARCH=%s", goos, goarch)
|
beeLogger.Log.Infof("Using: GOOS=%s GOARCH=%s", goos, goarch)
|
||||||
|
|
||||||
binPath := path.Join(tmpdir, appName)
|
binPath := path.Join(tmpdir, appName)
|
||||||
if goos == "windows" {
|
if goos == "windows" {
|
||||||
@ -538,10 +515,10 @@ func packApp(cmd *Command, args []string) int {
|
|||||||
execmd.Dir = thePath
|
execmd.Dir = thePath
|
||||||
err = execmd.Run()
|
err = execmd.Run()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatal(err.Error())
|
beeLogger.Log.Fatal(err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.Success("Build Successful!")
|
beeLogger.Log.Success("Build Successful!")
|
||||||
}
|
}
|
||||||
|
|
||||||
switch format {
|
switch format {
|
||||||
@ -552,14 +529,14 @@ func packApp(cmd *Command, args []string) int {
|
|||||||
|
|
||||||
outputN := appName + "." + format
|
outputN := appName + "." + format
|
||||||
|
|
||||||
if outputP == "" || path.IsAbs(outputP) == false {
|
if outputP == "" || !path.IsAbs(outputP) {
|
||||||
outputP = path.Join(curPath, outputP)
|
outputP = path.Join(curPath, outputP)
|
||||||
}
|
}
|
||||||
|
|
||||||
if _, err := os.Stat(outputP); err != nil {
|
if _, err := os.Stat(outputP); err != nil {
|
||||||
err = os.MkdirAll(outputP, 0755)
|
err = os.MkdirAll(outputP, 0755)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatal(err.Error())
|
beeLogger.Log.Fatal(err.Error())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -581,20 +558,20 @@ func packApp(cmd *Command, args []string) int {
|
|||||||
for _, r := range excludeR {
|
for _, r := range excludeR {
|
||||||
if len(r) > 0 {
|
if len(r) > 0 {
|
||||||
if re, err := regexp.Compile(r); err != nil {
|
if re, err := regexp.Compile(r); err != nil {
|
||||||
logger.Fatal(err.Error())
|
beeLogger.Log.Fatal(err.Error())
|
||||||
} else {
|
} else {
|
||||||
exr = append(exr, re)
|
exr = append(exr, re)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.Infof("Writing to output: %s", outputP)
|
beeLogger.Log.Infof("Writing to output: %s", outputP)
|
||||||
|
|
||||||
err = packDirectory(output, exp, exs, exr, tmpdir, thePath)
|
err = packDirectory(output, exp, exs, exr, tmpdir, thePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatal(err.Error())
|
beeLogger.Log.Fatal(err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.Success("Application packed!")
|
beeLogger.Log.Success("Application packed!")
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
102
cmd/commands/rs/rs.go
Normal file
102
cmd/commands/rs/rs.go
Normal file
@ -0,0 +1,102 @@
|
|||||||
|
// Copyright 2017 bee 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 rs ...
|
||||||
|
package rs
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"runtime"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
"github.com/beego/bee/cmd/commands/version"
|
||||||
|
"github.com/beego/bee/config"
|
||||||
|
"github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/logger/colors"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
|
)
|
||||||
|
|
||||||
|
var cmdRs = &commands.Command{
|
||||||
|
UsageLine: "rs",
|
||||||
|
Short: "Run customized scripts",
|
||||||
|
Long: `Run script allows you to run arbitrary commands using Bee.
|
||||||
|
Custom commands are provided from the "scripts" object inside bee.json or Beefile.
|
||||||
|
|
||||||
|
To run a custom command, use: {{"$ bee rs mycmd ARGS" | bold}}
|
||||||
|
{{if len .}}
|
||||||
|
{{"AVAILABLE SCRIPTS"|headline}}{{range $cmdName, $cmd := .}}
|
||||||
|
{{$cmdName | bold}}
|
||||||
|
{{$cmd}}{{end}}{{end}}
|
||||||
|
`,
|
||||||
|
PreRun: func(cmd *commands.Command, args []string) { version.ShowShortVersionBanner() },
|
||||||
|
Run: runScript,
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
config.LoadConfig()
|
||||||
|
cmdRs.Long = utils.TmplToString(cmdRs.Long, config.Conf.Scripts)
|
||||||
|
commands.AvailableCommands = append(commands.AvailableCommands, cmdRs)
|
||||||
|
}
|
||||||
|
|
||||||
|
func runScript(cmd *commands.Command, args []string) int {
|
||||||
|
if len(args) == 0 {
|
||||||
|
cmd.Usage()
|
||||||
|
}
|
||||||
|
|
||||||
|
start := time.Now()
|
||||||
|
script, args := args[0], args[1:]
|
||||||
|
|
||||||
|
if c, exist := config.Conf.Scripts[script]; exist {
|
||||||
|
command := customCommand{
|
||||||
|
Name: script,
|
||||||
|
Command: c,
|
||||||
|
Args: args,
|
||||||
|
}
|
||||||
|
if err := command.run(); err != nil {
|
||||||
|
beeLogger.Log.Error(err.Error())
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
beeLogger.Log.Errorf("Command '%s' not found in Beefile/bee.json", script)
|
||||||
|
}
|
||||||
|
elapsed := time.Since(start)
|
||||||
|
fmt.Println(colors.GreenBold(fmt.Sprintf("Finished in %s.", elapsed)))
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
type customCommand struct {
|
||||||
|
Name string
|
||||||
|
Command string
|
||||||
|
Args []string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *customCommand) run() error {
|
||||||
|
beeLogger.Log.Info(colors.GreenBold(fmt.Sprintf("Running '%s'...", c.Name)))
|
||||||
|
var cmd *exec.Cmd
|
||||||
|
switch runtime.GOOS {
|
||||||
|
case "darwin", "linux":
|
||||||
|
args := append([]string{c.Command}, c.Args...)
|
||||||
|
cmd = exec.Command("sh", "-c", strings.Join(args, " "))
|
||||||
|
case "windows":
|
||||||
|
args := append([]string{c.Command}, c.Args...)
|
||||||
|
cmd = exec.Command("cmd", "/C", strings.Join(args, " "))
|
||||||
|
}
|
||||||
|
cmd.Stdout = os.Stdout
|
||||||
|
cmd.Stderr = os.Stderr
|
||||||
|
return cmd.Run()
|
||||||
|
}
|
88
cmd/commands/run/docs.go
Normal file
88
cmd/commands/run/docs.go
Normal file
@ -0,0 +1,88 @@
|
|||||||
|
package run
|
||||||
|
|
||||||
|
import (
|
||||||
|
"archive/zip"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
swaggerVersion = "2"
|
||||||
|
swaggerlink = "https://github.com/beego/swagger/archive/v" + swaggerVersion + ".zip"
|
||||||
|
)
|
||||||
|
|
||||||
|
func downloadFromURL(url, fileName string) {
|
||||||
|
var down bool
|
||||||
|
if fd, err := os.Stat(fileName); err != nil && os.IsNotExist(err) {
|
||||||
|
down = true
|
||||||
|
} else if fd.Size() == int64(0) {
|
||||||
|
down = true
|
||||||
|
} else {
|
||||||
|
beeLogger.Log.Infof("'%s' already exists", fileName)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if down {
|
||||||
|
beeLogger.Log.Infof("Downloading '%s' to '%s'...", url, fileName)
|
||||||
|
output, err := os.Create(fileName)
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Errorf("Error while creating '%s': %s", fileName, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer output.Close()
|
||||||
|
|
||||||
|
response, err := http.Get(url)
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Errorf("Error while downloading '%s': %s", url, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer response.Body.Close()
|
||||||
|
|
||||||
|
n, err := io.Copy(output, response.Body)
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Errorf("Error while downloading '%s': %s", url, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
beeLogger.Log.Successf("%d bytes downloaded!", n)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func unzipAndDelete(src string) error {
|
||||||
|
beeLogger.Log.Infof("Unzipping '%s'...", src)
|
||||||
|
r, err := zip.OpenReader(src)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer r.Close()
|
||||||
|
|
||||||
|
rp := strings.NewReplacer("swagger-"+swaggerVersion, "swagger")
|
||||||
|
for _, f := range r.File {
|
||||||
|
rc, err := f.Open()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer rc.Close()
|
||||||
|
|
||||||
|
fname := rp.Replace(f.Name)
|
||||||
|
if f.FileInfo().IsDir() {
|
||||||
|
os.MkdirAll(fname, f.Mode())
|
||||||
|
} else {
|
||||||
|
f, err := os.OpenFile(
|
||||||
|
fname, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, f.Mode())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer f.Close()
|
||||||
|
|
||||||
|
_, err = io.Copy(f, rc)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
beeLogger.Log.Successf("Done! Deleting '%s'...", src)
|
||||||
|
return os.RemoveAll(src)
|
||||||
|
}
|
@ -11,13 +11,14 @@
|
|||||||
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
package main
|
package run
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"net/http"
|
"net/http"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
"github.com/gorilla/websocket"
|
"github.com/gorilla/websocket"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -70,7 +71,7 @@ func (c *wsClient) readPump() {
|
|||||||
_, _, err := c.conn.ReadMessage()
|
_, _, err := c.conn.ReadMessage()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if websocket.IsUnexpectedCloseError(err, websocket.CloseGoingAway) {
|
if websocket.IsUnexpectedCloseError(err, websocket.CloseGoingAway) {
|
||||||
logger.Errorf("An error happened when reading from the Websocket client: %v", err)
|
beeLogger.Log.Errorf("An error happened when reading from the Websocket client: %v", err)
|
||||||
}
|
}
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
@ -109,7 +110,7 @@ func (c *wsClient) writePump() {
|
|||||||
|
|
||||||
n := len(c.send)
|
n := len(c.send)
|
||||||
for i := 0; i < n; i++ {
|
for i := 0; i < n; i++ {
|
||||||
w.Write(newline)
|
w.Write([]byte("/n"))
|
||||||
w.Write(<-c.send)
|
w.Write(<-c.send)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -155,13 +156,13 @@ func startReloadServer() {
|
|||||||
})
|
})
|
||||||
|
|
||||||
go startServer()
|
go startServer()
|
||||||
logger.Infof("Reload server listening at %s", reloadAddress)
|
beeLogger.Log.Infof("Reload server listening at %s", reloadAddress)
|
||||||
}
|
}
|
||||||
|
|
||||||
func startServer() {
|
func startServer() {
|
||||||
err := http.ListenAndServe(reloadAddress, nil)
|
err := http.ListenAndServe(reloadAddress, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Errorf("Failed to start up the Reload server: %v", err)
|
beeLogger.Log.Errorf("Failed to start up the Reload server: %v", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -175,7 +176,7 @@ func sendReload(payload string) {
|
|||||||
func handleWsRequest(broker *wsBroker, w http.ResponseWriter, r *http.Request) {
|
func handleWsRequest(broker *wsBroker, w http.ResponseWriter, r *http.Request) {
|
||||||
conn, err := upgrader.Upgrade(w, r, nil)
|
conn, err := upgrader.Upgrade(w, r, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Errorf("error while upgrading server connection: %v", err)
|
beeLogger.Log.Errorf("error while upgrading server connection: %v", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
@ -11,8 +11,7 @@
|
|||||||
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
package run
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
@ -20,25 +19,31 @@ import (
|
|||||||
path "path/filepath"
|
path "path/filepath"
|
||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
"github.com/beego/bee/cmd/commands/version"
|
||||||
|
"github.com/beego/bee/config"
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
var cmdRun = &Command{
|
var CmdRun = &commands.Command{
|
||||||
UsageLine: "run [appname] [watchall] [-main=*.go] [-downdoc=true] [-gendoc=true] [-vendor=true] [-e=folderToExclude] [-tags=goBuildTags] [-runmode=BEEGO_RUNMODE]",
|
UsageLine: "run [appname] [watchall] [-main=*.go] [-downdoc=true] [-gendoc=true] [-vendor=true] [-e=folderToExclude] [-ex=extraPackageToWatch] [-tags=goBuildTags] [-runmode=BEEGO_RUNMODE]",
|
||||||
Short: "Run the application by starting a local development server",
|
Short: "Run the application by starting a local development server",
|
||||||
Long: `
|
Long: `
|
||||||
Run command will supervise the filesystem of the application for any changes, and recompile/restart it.
|
Run command will supervise the filesystem of the application for any changes, and recompile/restart it.
|
||||||
|
|
||||||
`,
|
`,
|
||||||
PreRun: func(cmd *Command, args []string) { ShowShortVersionBanner() },
|
PreRun: func(cmd *commands.Command, args []string) { version.ShowShortVersionBanner() },
|
||||||
Run: runApp,
|
Run: RunApp,
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
mainFiles ListOpts
|
mainFiles utils.ListOpts
|
||||||
downdoc docValue
|
downdoc utils.DocValue
|
||||||
gendoc docValue
|
gendoc utils.DocValue
|
||||||
// The flags list of the paths excluded from watching
|
// The flags list of the paths excluded from watching
|
||||||
excludedPaths strFlags
|
excludedPaths utils.StrFlags
|
||||||
// Pass through to -tags arg of "go build"
|
// Pass through to -tags arg of "go build"
|
||||||
buildTags string
|
buildTags string
|
||||||
// Application path
|
// Application path
|
||||||
@ -53,65 +58,64 @@ var (
|
|||||||
currentGoPath string
|
currentGoPath string
|
||||||
// Current runmode
|
// Current runmode
|
||||||
runmode string
|
runmode string
|
||||||
|
// Extra directories
|
||||||
|
extraPackages utils.StrFlags
|
||||||
)
|
)
|
||||||
|
var started = make(chan bool)
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
cmdRun.Flag.Var(&mainFiles, "main", "Specify main go files.")
|
CmdRun.Flag.Var(&mainFiles, "main", "Specify main go files.")
|
||||||
cmdRun.Flag.Var(&gendoc, "gendoc", "Enable auto-generate the docs.")
|
CmdRun.Flag.Var(&gendoc, "gendoc", "Enable auto-generate the docs.")
|
||||||
cmdRun.Flag.Var(&downdoc, "downdoc", "Enable auto-download of the swagger file if it does not exist.")
|
CmdRun.Flag.Var(&downdoc, "downdoc", "Enable auto-download of the swagger file if it does not exist.")
|
||||||
cmdRun.Flag.Var(&excludedPaths, "e", "List of paths to exclude.")
|
CmdRun.Flag.Var(&excludedPaths, "e", "List of paths to exclude.")
|
||||||
cmdRun.Flag.BoolVar(&vendorWatch, "vendor", false, "Enable watch vendor folder.")
|
CmdRun.Flag.BoolVar(&vendorWatch, "vendor", false, "Enable watch vendor folder.")
|
||||||
cmdRun.Flag.StringVar(&buildTags, "tags", "", "Set the build tags. See: https://golang.org/pkg/go/build/")
|
CmdRun.Flag.StringVar(&buildTags, "tags", "", "Set the build tags. See: https://golang.org/pkg/go/build/")
|
||||||
cmdRun.Flag.StringVar(&runmode, "runmode", "", "Set the Beego run mode.")
|
CmdRun.Flag.StringVar(&runmode, "runmode", "", "Set the Beego run mode.")
|
||||||
|
CmdRun.Flag.Var(&extraPackages, "ex", "List of extra package to watch.")
|
||||||
exit = make(chan bool)
|
exit = make(chan bool)
|
||||||
|
commands.AvailableCommands = append(commands.AvailableCommands, CmdRun)
|
||||||
}
|
}
|
||||||
|
|
||||||
func runApp(cmd *Command, args []string) int {
|
func RunApp(cmd *commands.Command, args []string) int {
|
||||||
if len(args) == 0 || args[0] == "watchall" {
|
if len(args) == 0 || args[0] == "watchall" {
|
||||||
currpath, _ = os.Getwd()
|
currpath, _ = os.Getwd()
|
||||||
|
if found, _gopath, _ := utils.SearchGOPATHs(currpath); found {
|
||||||
if found, _gopath, _ := SearchGOPATHs(currpath); found {
|
|
||||||
appname = path.Base(currpath)
|
appname = path.Base(currpath)
|
||||||
currentGoPath = _gopath
|
currentGoPath = _gopath
|
||||||
} else {
|
} else {
|
||||||
logger.Fatalf("No application '%s' found in your GOPATH", currpath)
|
beeLogger.Log.Fatalf("No application '%s' found in your GOPATH", currpath)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Check if passed Bee application path/name exists in the GOPATH(s)
|
// Check if passed Bee application path/name exists in the GOPATH(s)
|
||||||
if found, _gopath, _path := SearchGOPATHs(args[0]); found {
|
if found, _gopath, _path := utils.SearchGOPATHs(args[0]); found {
|
||||||
currpath = _path
|
currpath = _path
|
||||||
currentGoPath = _gopath
|
currentGoPath = _gopath
|
||||||
appname = path.Base(currpath)
|
appname = path.Base(currpath)
|
||||||
} else {
|
} else {
|
||||||
logger.Fatalf("No application '%s' found in your GOPATH", args[0])
|
beeLogger.Log.Fatalf("No application '%s' found in your GOPATH", args[0])
|
||||||
}
|
}
|
||||||
|
|
||||||
if strings.HasSuffix(appname, ".go") && isExist(currpath) {
|
if strings.HasSuffix(appname, ".go") && utils.IsExist(currpath) {
|
||||||
logger.Warnf("The appname is in conflict with file's current path. Do you want to build appname as '%s'", appname)
|
beeLogger.Log.Warnf("The appname is in conflict with file's current path. Do you want to build appname as '%s'", appname)
|
||||||
logger.Info("Do you want to overwrite it? [yes|no] ")
|
beeLogger.Log.Info("Do you want to overwrite it? [yes|no] ")
|
||||||
if !askForConfirmation() {
|
if !utils.AskForConfirmation() {
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.Infof("Using '%s' as 'appname'", appname)
|
beeLogger.Log.Infof("Using '%s' as 'appname'", appname)
|
||||||
|
|
||||||
logger.Debugf("Current path: %s", __FILE__(), __LINE__(), currpath)
|
beeLogger.Log.Debugf("Current path: %s", utils.FILE(), utils.LINE(), currpath)
|
||||||
|
|
||||||
if runmode == "prod" || runmode == "dev" {
|
if runmode == "prod" || runmode == "dev" {
|
||||||
os.Setenv("BEEGO_RUNMODE", runmode)
|
os.Setenv("BEEGO_RUNMODE", runmode)
|
||||||
logger.Infof("Using '%s' as 'runmode'", os.Getenv("BEEGO_RUNMODE"))
|
beeLogger.Log.Infof("Using '%s' as 'runmode'", os.Getenv("BEEGO_RUNMODE"))
|
||||||
} else if runmode != "" {
|
} else if runmode != "" {
|
||||||
os.Setenv("BEEGO_RUNMODE", runmode)
|
os.Setenv("BEEGO_RUNMODE", runmode)
|
||||||
logger.Warnf("Using '%s' as 'runmode'", os.Getenv("BEEGO_RUNMODE"))
|
beeLogger.Log.Warnf("Using '%s' as 'runmode'", os.Getenv("BEEGO_RUNMODE"))
|
||||||
} else if os.Getenv("BEEGO_RUNMODE") != "" {
|
} else if os.Getenv("BEEGO_RUNMODE") != "" {
|
||||||
logger.Warnf("Using '%s' as 'runmode'", os.Getenv("BEEGO_RUNMODE"))
|
beeLogger.Log.Warnf("Using '%s' as 'runmode'", os.Getenv("BEEGO_RUNMODE"))
|
||||||
}
|
|
||||||
|
|
||||||
err := loadConfig()
|
|
||||||
if err != nil {
|
|
||||||
logger.Fatalf("Failed to load configuration: %s", err)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var paths []string
|
var paths []string
|
||||||
@ -119,10 +123,32 @@ func runApp(cmd *Command, args []string) int {
|
|||||||
|
|
||||||
// Because monitor files has some issues, we watch current directory
|
// Because monitor files has some issues, we watch current directory
|
||||||
// and ignore non-go files.
|
// and ignore non-go files.
|
||||||
for _, p := range conf.DirStruct.Others {
|
for _, p := range config.Conf.DirStruct.Others {
|
||||||
paths = append(paths, strings.Replace(p, "$GOPATH", currentGoPath, -1))
|
paths = append(paths, strings.Replace(p, "$GOPATH", currentGoPath, -1))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if len(extraPackages) > 0 {
|
||||||
|
// get the full path
|
||||||
|
for _, packagePath := range extraPackages {
|
||||||
|
if found, _, _fullPath := utils.SearchGOPATHs(packagePath); found {
|
||||||
|
readAppDirectories(_fullPath, &paths)
|
||||||
|
} else {
|
||||||
|
beeLogger.Log.Warnf("No extra package '%s' found in your GOPATH", packagePath)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// let paths unique
|
||||||
|
strSet := make(map[string]struct{})
|
||||||
|
for _, p := range paths {
|
||||||
|
strSet[p] = struct{}{}
|
||||||
|
}
|
||||||
|
paths = make([]string, len(strSet))
|
||||||
|
index := 0
|
||||||
|
for i := range strSet {
|
||||||
|
paths[index] = i
|
||||||
|
index++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
files := []string{}
|
files := []string{}
|
||||||
for _, arg := range mainFiles {
|
for _, arg := range mainFiles {
|
||||||
if len(arg) > 0 {
|
if len(arg) > 0 {
|
||||||
@ -139,7 +165,7 @@ func runApp(cmd *Command, args []string) int {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Start the Reload server (if enabled)
|
// Start the Reload server (if enabled)
|
||||||
if conf.EnableReload {
|
if config.Conf.EnableReload {
|
||||||
startReloadServer()
|
startReloadServer()
|
||||||
}
|
}
|
||||||
if gendoc == "true" {
|
if gendoc == "true" {
|
||||||
@ -151,10 +177,8 @@ func runApp(cmd *Command, args []string) int {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for {
|
for {
|
||||||
select {
|
<-exit
|
||||||
case <-exit:
|
runtime.Goexit()
|
||||||
runtime.Goexit()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -181,16 +205,16 @@ func readAppDirectories(directory string, paths *[]string) {
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
if fileInfo.IsDir() == true && fileInfo.Name()[0] != '.' {
|
if fileInfo.IsDir() && fileInfo.Name()[0] != '.' {
|
||||||
readAppDirectories(directory+"/"+fileInfo.Name(), paths)
|
readAppDirectories(directory+"/"+fileInfo.Name(), paths)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
if useDirectory == true {
|
if useDirectory {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
if path.Ext(fileInfo.Name()) == ".go" || (ifStaticFile(fileInfo.Name()) && conf.EnableReload) {
|
if path.Ext(fileInfo.Name()) == ".go" || (ifStaticFile(fileInfo.Name()) && config.Conf.EnableReload) {
|
||||||
*paths = append(*paths, directory)
|
*paths = append(*paths, directory)
|
||||||
useDirectory = true
|
useDirectory = true
|
||||||
}
|
}
|
||||||
@ -203,16 +227,16 @@ func isExcluded(filePath string) bool {
|
|||||||
for _, p := range excludedPaths {
|
for _, p := range excludedPaths {
|
||||||
absP, err := path.Abs(p)
|
absP, err := path.Abs(p)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Errorf("Cannot get absolute path of '%s'", p)
|
beeLogger.Log.Errorf("Cannot get absolute path of '%s'", p)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
absFilePath, err := path.Abs(filePath)
|
absFilePath, err := path.Abs(filePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Errorf("Cannot get absolute path of '%s'", filePath)
|
beeLogger.Log.Errorf("Cannot get absolute path of '%s'", filePath)
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
if strings.HasPrefix(absFilePath, absP) {
|
if strings.HasPrefix(absFilePath, absP) {
|
||||||
logger.Infof("'%s' is not being watched", filePath)
|
beeLogger.Log.Infof("'%s' is not being watched", filePath)
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -12,7 +12,7 @@
|
|||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
|
||||||
package main
|
package run
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
@ -24,6 +24,10 @@ import (
|
|||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/beego/bee/config"
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/logger/colors"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
"github.com/fsnotify/fsnotify"
|
"github.com/fsnotify/fsnotify"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -46,7 +50,7 @@ var (
|
|||||||
func NewWatcher(paths []string, files []string, isgenerate bool) {
|
func NewWatcher(paths []string, files []string, isgenerate bool) {
|
||||||
watcher, err := fsnotify.NewWatcher()
|
watcher, err := fsnotify.NewWatcher()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Failed to create watcher: %s", err)
|
beeLogger.Log.Fatalf("Failed to create watcher: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
@ -55,7 +59,7 @@ func NewWatcher(paths []string, files []string, isgenerate bool) {
|
|||||||
case e := <-watcher.Events:
|
case e := <-watcher.Events:
|
||||||
isBuild := true
|
isBuild := true
|
||||||
|
|
||||||
if ifStaticFile(e.Name) && conf.EnableReload {
|
if ifStaticFile(e.Name) && config.Conf.EnableReload {
|
||||||
sendReload(e.String())
|
sendReload(e.String())
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
@ -69,14 +73,14 @@ func NewWatcher(paths []string, files []string, isgenerate bool) {
|
|||||||
|
|
||||||
mt := getFileModTime(e.Name)
|
mt := getFileModTime(e.Name)
|
||||||
if t := eventTime[e.Name]; mt == t {
|
if t := eventTime[e.Name]; mt == t {
|
||||||
logger.Infof(bold("Skipping: ")+"%s", e.String())
|
beeLogger.Log.Hintf(colors.Bold("Skipping: ")+"%s", e.String())
|
||||||
isBuild = false
|
isBuild = false
|
||||||
}
|
}
|
||||||
|
|
||||||
eventTime[e.Name] = mt
|
eventTime[e.Name] = mt
|
||||||
|
|
||||||
if isBuild {
|
if isBuild {
|
||||||
logger.Infof("Event fired: %s", e)
|
beeLogger.Log.Hintf("Event fired: %s", e)
|
||||||
go func() {
|
go func() {
|
||||||
// Wait 1s before autobuild until there is no file change.
|
// Wait 1s before autobuild until there is no file change.
|
||||||
scheduleTime = time.Now().Add(1 * time.Second)
|
scheduleTime = time.Now().Add(1 * time.Second)
|
||||||
@ -85,17 +89,17 @@ func NewWatcher(paths []string, files []string, isgenerate bool) {
|
|||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
case err := <-watcher.Errors:
|
case err := <-watcher.Errors:
|
||||||
logger.Warnf("Watcher error: %s", err.Error()) // No need to exit here
|
beeLogger.Log.Warnf("Watcher error: %s", err.Error()) // No need to exit here
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
logger.Info("Initializing watcher...")
|
beeLogger.Log.Info("Initializing watcher...")
|
||||||
for _, path := range paths {
|
for _, path := range paths {
|
||||||
logger.Infof(bold("Watching: ")+"%s", path)
|
beeLogger.Log.Hintf(colors.Bold("Watching: ")+"%s", path)
|
||||||
err = watcher.Add(path)
|
err = watcher.Add(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Failed to watch directory: %s", err)
|
beeLogger.Log.Fatalf("Failed to watch directory: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -105,14 +109,14 @@ func getFileModTime(path string) int64 {
|
|||||||
path = strings.Replace(path, "\\", "/", -1)
|
path = strings.Replace(path, "\\", "/", -1)
|
||||||
f, err := os.Open(path)
|
f, err := os.Open(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Errorf("Failed to open file on '%s': %s", path, err)
|
beeLogger.Log.Errorf("Failed to open file on '%s': %s", path, err)
|
||||||
return time.Now().Unix()
|
return time.Now().Unix()
|
||||||
}
|
}
|
||||||
defer f.Close()
|
defer f.Close()
|
||||||
|
|
||||||
fi, err := f.Stat()
|
fi, err := f.Stat()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Errorf("Failed to get file stats: %s", err)
|
beeLogger.Log.Errorf("Failed to get file stats: %s", err)
|
||||||
return time.Now().Unix()
|
return time.Now().Unix()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -127,57 +131,32 @@ func AutoBuild(files []string, isgenerate bool) {
|
|||||||
os.Chdir(currpath)
|
os.Chdir(currpath)
|
||||||
|
|
||||||
cmdName := "go"
|
cmdName := "go"
|
||||||
if conf.Gopm.Enable {
|
|
||||||
cmdName = "gopm"
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
var (
|
||||||
err error
|
err error
|
||||||
stderr bytes.Buffer
|
stderr bytes.Buffer
|
||||||
stdout bytes.Buffer
|
|
||||||
)
|
)
|
||||||
// For applications use full import path like "github.com/.../.."
|
// For applications use full import path like "github.com/.../.."
|
||||||
// are able to use "go install" to reduce build time.
|
// are able to use "go install" to reduce build time.
|
||||||
if conf.GoInstall {
|
if config.Conf.GoInstall {
|
||||||
icmd := exec.Command(cmdName, "install", "-v")
|
icmd := exec.Command(cmdName, "install", "-v")
|
||||||
icmd.Stdout = os.Stdout
|
icmd.Stdout = os.Stdout
|
||||||
icmd.Stderr = os.Stderr
|
icmd.Stderr = os.Stderr
|
||||||
icmd.Env = append(os.Environ(), "GOGC=off")
|
icmd.Env = append(os.Environ(), "GOGC=off")
|
||||||
icmd.Run()
|
icmd.Run()
|
||||||
}
|
}
|
||||||
if conf.Gopm.Install {
|
|
||||||
icmd := exec.Command("go", "list", "./...")
|
|
||||||
icmd.Stdout = &stdout
|
|
||||||
icmd.Env = append(os.Environ(), "GOGC=off")
|
|
||||||
err = icmd.Run()
|
|
||||||
if err == nil {
|
|
||||||
list := strings.Split(stdout.String(), "\n")[1:]
|
|
||||||
for _, pkg := range list {
|
|
||||||
if len(pkg) == 0 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
icmd = exec.Command(cmdName, "install", pkg)
|
|
||||||
icmd.Stdout = os.Stdout
|
|
||||||
icmd.Stderr = os.Stderr
|
|
||||||
icmd.Env = append(os.Environ(), "GOGC=off")
|
|
||||||
err = icmd.Run()
|
|
||||||
if err != nil {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if isgenerate {
|
if isgenerate {
|
||||||
logger.Info("Generating the docs...")
|
beeLogger.Log.Info("Generating the docs...")
|
||||||
icmd := exec.Command("bee", "generate", "docs")
|
icmd := exec.Command("bee", "generate", "docs")
|
||||||
icmd.Env = append(os.Environ(), "GOGC=off")
|
icmd.Env = append(os.Environ(), "GOGC=off")
|
||||||
err = icmd.Run()
|
err = icmd.Run()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Errorf("Failed to generate the docs.")
|
utils.Notify("", "Failed to generate the docs.")
|
||||||
|
beeLogger.Log.Errorf("Failed to generate the docs.")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
logger.Success("Docs generated!")
|
beeLogger.Log.Success("Docs generated!")
|
||||||
}
|
}
|
||||||
|
|
||||||
if err == nil {
|
if err == nil {
|
||||||
@ -198,12 +177,13 @@ func AutoBuild(files []string, isgenerate bool) {
|
|||||||
bcmd.Stderr = &stderr
|
bcmd.Stderr = &stderr
|
||||||
err = bcmd.Run()
|
err = bcmd.Run()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Errorf("Failed to build the application: %s", stderr.String())
|
utils.Notify(stderr.String(), "Build Failed")
|
||||||
|
beeLogger.Log.Errorf("Failed to build the application: %s", stderr.String())
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.Success("Built Successfully!")
|
beeLogger.Log.Success("Built Successfully!")
|
||||||
Restart(appname)
|
Restart(appname)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -211,39 +191,39 @@ func AutoBuild(files []string, isgenerate bool) {
|
|||||||
func Kill() {
|
func Kill() {
|
||||||
defer func() {
|
defer func() {
|
||||||
if e := recover(); e != nil {
|
if e := recover(); e != nil {
|
||||||
logger.Infof("Kill recover: %s", e)
|
beeLogger.Log.Infof("Kill recover: %s", e)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
if cmd != nil && cmd.Process != nil {
|
if cmd != nil && cmd.Process != nil {
|
||||||
err := cmd.Process.Kill()
|
err := cmd.Process.Kill()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Errorf("Error while killing cmd process: %s", err)
|
beeLogger.Log.Errorf("Error while killing cmd process: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Restart kills the running command process and starts it again
|
// Restart kills the running command process and starts it again
|
||||||
func Restart(appname string) {
|
func Restart(appname string) {
|
||||||
logger.Debugf("Kill running process", __FILE__(), __LINE__())
|
beeLogger.Log.Debugf("Kill running process", utils.FILE(), utils.LINE())
|
||||||
Kill()
|
Kill()
|
||||||
go Start(appname)
|
go Start(appname)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Start starts the command process
|
// Start starts the command process
|
||||||
func Start(appname string) {
|
func Start(appname string) {
|
||||||
logger.Infof("Restarting '%s'...", appname)
|
beeLogger.Log.Infof("Restarting '%s'...", appname)
|
||||||
if strings.Index(appname, "./") == -1 {
|
if !strings.Contains(appname, "./") {
|
||||||
appname = "./" + appname
|
appname = "./" + appname
|
||||||
}
|
}
|
||||||
|
|
||||||
cmd = exec.Command(appname)
|
cmd = exec.Command(appname)
|
||||||
cmd.Stdout = os.Stdout
|
cmd.Stdout = os.Stdout
|
||||||
cmd.Stderr = os.Stderr
|
cmd.Stderr = os.Stderr
|
||||||
cmd.Args = append([]string{appname}, conf.CmdArgs...)
|
cmd.Args = append([]string{appname}, config.Conf.CmdArgs...)
|
||||||
cmd.Env = append(os.Environ(), conf.Envs...)
|
cmd.Env = append(os.Environ(), config.Conf.Envs...)
|
||||||
|
|
||||||
go cmd.Run()
|
go cmd.Run()
|
||||||
logger.Successf("'%s' is running...", appname)
|
beeLogger.Log.Successf("'%s' is running...", appname)
|
||||||
started <- true
|
started <- true
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -262,7 +242,7 @@ func shouldIgnoreFile(filename string) bool {
|
|||||||
for _, regex := range ignoredFilesRegExps {
|
for _, regex := range ignoredFilesRegExps {
|
||||||
r, err := regexp.Compile(regex)
|
r, err := regexp.Compile(regex)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Could not compile regular expression: %s", err)
|
beeLogger.Log.Fatalf("Could not compile regular expression: %s", err)
|
||||||
}
|
}
|
||||||
if r.MatchString(filename) {
|
if r.MatchString(filename) {
|
||||||
return true
|
return true
|
@ -1,4 +1,4 @@
|
|||||||
package main
|
package version
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"io"
|
"io"
|
||||||
@ -6,6 +6,10 @@ import (
|
|||||||
"os"
|
"os"
|
||||||
"runtime"
|
"runtime"
|
||||||
"text/template"
|
"text/template"
|
||||||
|
|
||||||
|
"time"
|
||||||
|
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
// RuntimeInfo holds information about the current runtime.
|
// RuntimeInfo holds information about the current runtime.
|
||||||
@ -26,12 +30,12 @@ type RuntimeInfo struct {
|
|||||||
// print the banner in case of error.
|
// print the banner in case of error.
|
||||||
func InitBanner(out io.Writer, in io.Reader) {
|
func InitBanner(out io.Writer, in io.Reader) {
|
||||||
if in == nil {
|
if in == nil {
|
||||||
logger.Fatal("The input is nil")
|
beeLogger.Log.Fatal("The input is nil")
|
||||||
}
|
}
|
||||||
|
|
||||||
banner, err := ioutil.ReadAll(in)
|
banner, err := ioutil.ReadAll(in)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Error while trying to read the banner: %s", err)
|
beeLogger.Log.Fatalf("Error while trying to read the banner: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
show(out, string(banner))
|
show(out, string(banner))
|
||||||
@ -43,11 +47,11 @@ func show(out io.Writer, content string) {
|
|||||||
Parse(content)
|
Parse(content)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Cannot parse the banner template: %s", err)
|
beeLogger.Log.Fatalf("Cannot parse the banner template: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
err = t.Execute(out, RuntimeInfo{
|
err = t.Execute(out, RuntimeInfo{
|
||||||
getGoVersion(),
|
GetGoVersion(),
|
||||||
runtime.GOOS,
|
runtime.GOOS,
|
||||||
runtime.GOARCH,
|
runtime.GOARCH,
|
||||||
runtime.NumCPU(),
|
runtime.NumCPU(),
|
||||||
@ -55,7 +59,14 @@ func show(out io.Writer, content string) {
|
|||||||
runtime.GOROOT(),
|
runtime.GOROOT(),
|
||||||
runtime.Compiler,
|
runtime.Compiler,
|
||||||
version,
|
version,
|
||||||
getBeegoVersion(),
|
GetBeegoVersion(),
|
||||||
})
|
})
|
||||||
MustCheck(err)
|
if err != nil {
|
||||||
|
beeLogger.Log.Error(err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now returns the current local time in the specified layout
|
||||||
|
func Now(layout string) string {
|
||||||
|
return time.Now().Format(layout)
|
||||||
}
|
}
|
@ -1,4 +1,4 @@
|
|||||||
package main
|
package version
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
@ -14,18 +14,12 @@ import (
|
|||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/logger/colors"
|
||||||
"gopkg.in/yaml.v2"
|
"gopkg.in/yaml.v2"
|
||||||
)
|
)
|
||||||
|
|
||||||
var cmdVersion = &Command{
|
|
||||||
UsageLine: "version",
|
|
||||||
Short: "Prints the current Bee version",
|
|
||||||
Long: `
|
|
||||||
Prints the current Bee, Beego and Go version alongside the platform information.
|
|
||||||
`,
|
|
||||||
Run: versionCmd,
|
|
||||||
}
|
|
||||||
|
|
||||||
const verboseVersionBanner string = `%s%s______
|
const verboseVersionBanner string = `%s%s______
|
||||||
| ___ \
|
| ___ \
|
||||||
| |_/ / ___ ___
|
| |_/ / ___ ___
|
||||||
@ -52,21 +46,33 @@ const shortVersionBanner = `______
|
|||||||
\____/ \___| \___| v{{ .BeeVersion }}
|
\____/ \___| \___| v{{ .BeeVersion }}
|
||||||
`
|
`
|
||||||
|
|
||||||
|
var CmdVersion = &commands.Command{
|
||||||
|
UsageLine: "version",
|
||||||
|
Short: "Prints the current Bee version",
|
||||||
|
Long: `
|
||||||
|
Prints the current Bee, Beego and Go version alongside the platform information.
|
||||||
|
`,
|
||||||
|
Run: versionCmd,
|
||||||
|
}
|
||||||
var outputFormat string
|
var outputFormat string
|
||||||
|
|
||||||
|
const version = "1.8.1"
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
fs := flag.NewFlagSet("version", flag.ContinueOnError)
|
fs := flag.NewFlagSet("version", flag.ContinueOnError)
|
||||||
fs.StringVar(&outputFormat, "o", "", "Set the output format. Either json or yaml.")
|
fs.StringVar(&outputFormat, "o", "", "Set the output format. Either json or yaml.")
|
||||||
cmdVersion.Flag = *fs
|
CmdVersion.Flag = *fs
|
||||||
|
commands.AvailableCommands = append(commands.AvailableCommands, CmdVersion)
|
||||||
}
|
}
|
||||||
|
|
||||||
func versionCmd(cmd *Command, args []string) int {
|
func versionCmd(cmd *commands.Command, args []string) int {
|
||||||
|
|
||||||
cmd.Flag.Parse(args)
|
cmd.Flag.Parse(args)
|
||||||
stdout := cmd.Out()
|
stdout := cmd.Out()
|
||||||
|
|
||||||
if outputFormat != "" {
|
if outputFormat != "" {
|
||||||
runtimeInfo := RuntimeInfo{
|
runtimeInfo := RuntimeInfo{
|
||||||
getGoVersion(),
|
GetGoVersion(),
|
||||||
runtime.GOOS,
|
runtime.GOOS,
|
||||||
runtime.GOARCH,
|
runtime.GOARCH,
|
||||||
runtime.NumCPU(),
|
runtime.NumCPU(),
|
||||||
@ -74,20 +80,24 @@ func versionCmd(cmd *Command, args []string) int {
|
|||||||
runtime.GOROOT(),
|
runtime.GOROOT(),
|
||||||
runtime.Compiler,
|
runtime.Compiler,
|
||||||
version,
|
version,
|
||||||
getBeegoVersion(),
|
GetBeegoVersion(),
|
||||||
}
|
}
|
||||||
switch outputFormat {
|
switch outputFormat {
|
||||||
case "json":
|
case "json":
|
||||||
{
|
{
|
||||||
b, err := json.MarshalIndent(runtimeInfo, "", " ")
|
b, err := json.MarshalIndent(runtimeInfo, "", " ")
|
||||||
MustCheck(err)
|
if err != nil {
|
||||||
|
beeLogger.Log.Error(err.Error())
|
||||||
|
}
|
||||||
fmt.Println(string(b))
|
fmt.Println(string(b))
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
case "yaml":
|
case "yaml":
|
||||||
{
|
{
|
||||||
b, err := yaml.Marshal(&runtimeInfo)
|
b, err := yaml.Marshal(&runtimeInfo)
|
||||||
MustCheck(err)
|
if err != nil {
|
||||||
|
beeLogger.Log.Error(err.Error())
|
||||||
|
}
|
||||||
fmt.Println(string(b))
|
fmt.Println(string(b))
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
@ -102,18 +112,18 @@ func versionCmd(cmd *Command, args []string) int {
|
|||||||
|
|
||||||
// ShowShortVersionBanner prints the short version banner.
|
// ShowShortVersionBanner prints the short version banner.
|
||||||
func ShowShortVersionBanner() {
|
func ShowShortVersionBanner() {
|
||||||
output := NewColorWriter(os.Stdout)
|
output := colors.NewColorWriter(os.Stdout)
|
||||||
InitBanner(output, bytes.NewBufferString(MagentaBold(shortVersionBanner)))
|
InitBanner(output, bytes.NewBufferString(colors.MagentaBold(shortVersionBanner)))
|
||||||
}
|
}
|
||||||
|
|
||||||
func getBeegoVersion() string {
|
func GetBeegoVersion() string {
|
||||||
gopath := os.Getenv("GOPATH")
|
gopath := os.Getenv("GOPATH")
|
||||||
re, err := regexp.Compile(`VERSION = "([0-9.]+)"`)
|
re, err := regexp.Compile(`VERSION = "([0-9.]+)"`)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
if gopath == "" {
|
if gopath == "" {
|
||||||
err = fmt.Errorf("You need to set GOPATH environment variable")
|
beeLogger.Log.Error("You need to set GOPATH environment variable")
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
wgopath := path.SplitList(gopath)
|
wgopath := path.SplitList(gopath)
|
||||||
@ -125,11 +135,11 @@ func getBeegoVersion() string {
|
|||||||
if os.IsNotExist(err) {
|
if os.IsNotExist(err) {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
logger.Error("Error while getting stats of 'beego.go'")
|
beeLogger.Log.Error("Error while getting stats of 'beego.go'")
|
||||||
}
|
}
|
||||||
fd, err := os.Open(filename)
|
fd, err := os.Open(filename)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Error("Error while reading 'beego.go'")
|
beeLogger.Log.Error("Error while reading 'beego.go'")
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
reader := bufio.NewReader(fd)
|
reader := bufio.NewReader(fd)
|
||||||
@ -152,14 +162,14 @@ func getBeegoVersion() string {
|
|||||||
return "Beego is not installed. Please do consider installing it first: https://github.com/astaxie/beego"
|
return "Beego is not installed. Please do consider installing it first: https://github.com/astaxie/beego"
|
||||||
}
|
}
|
||||||
|
|
||||||
func getGoVersion() string {
|
func GetGoVersion() string {
|
||||||
var (
|
var (
|
||||||
cmdOut []byte
|
cmdOut []byte
|
||||||
err error
|
err error
|
||||||
)
|
)
|
||||||
|
|
||||||
if cmdOut, err = exec.Command("go", "version").Output(); err != nil {
|
if cmdOut, err = exec.Command("go", "version").Output(); err != nil {
|
||||||
logger.Fatalf("There was an error running 'go version' command: %s", err)
|
beeLogger.Log.Fatalf("There was an error running 'go version' command: %s", err)
|
||||||
}
|
}
|
||||||
return strings.Split(string(cmdOut), " ")[2]
|
return strings.Split(string(cmdOut), " ")[2]
|
||||||
}
|
}
|
261
code.go
261
code.go
@ -1,261 +0,0 @@
|
|||||||
// Copyright 2011 Gary Burd
|
|
||||||
// Copyright 2013 Unknown
|
|
||||||
//
|
|
||||||
// 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 main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"go/ast"
|
|
||||||
"go/printer"
|
|
||||||
"go/scanner"
|
|
||||||
"go/token"
|
|
||||||
"math"
|
|
||||||
"strconv"
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
notPredeclared = iota
|
|
||||||
predeclaredType
|
|
||||||
predeclaredConstant
|
|
||||||
predeclaredFunction
|
|
||||||
)
|
|
||||||
|
|
||||||
// predeclared represents the set of all predeclared identifiers.
|
|
||||||
var predeclared = map[string]int{
|
|
||||||
"bool": predeclaredType,
|
|
||||||
"byte": predeclaredType,
|
|
||||||
"complex128": predeclaredType,
|
|
||||||
"complex64": predeclaredType,
|
|
||||||
"error": predeclaredType,
|
|
||||||
"float32": predeclaredType,
|
|
||||||
"float64": predeclaredType,
|
|
||||||
"int16": predeclaredType,
|
|
||||||
"int32": predeclaredType,
|
|
||||||
"int64": predeclaredType,
|
|
||||||
"int8": predeclaredType,
|
|
||||||
"int": predeclaredType,
|
|
||||||
"rune": predeclaredType,
|
|
||||||
"string": predeclaredType,
|
|
||||||
"uint16": predeclaredType,
|
|
||||||
"uint32": predeclaredType,
|
|
||||||
"uint64": predeclaredType,
|
|
||||||
"uint8": predeclaredType,
|
|
||||||
"uint": predeclaredType,
|
|
||||||
"uintptr": predeclaredType,
|
|
||||||
|
|
||||||
"true": predeclaredConstant,
|
|
||||||
"false": predeclaredConstant,
|
|
||||||
"iota": predeclaredConstant,
|
|
||||||
"nil": predeclaredConstant,
|
|
||||||
|
|
||||||
"append": predeclaredFunction,
|
|
||||||
"cap": predeclaredFunction,
|
|
||||||
"close": predeclaredFunction,
|
|
||||||
"complex": predeclaredFunction,
|
|
||||||
"copy": predeclaredFunction,
|
|
||||||
"delete": predeclaredFunction,
|
|
||||||
"imag": predeclaredFunction,
|
|
||||||
"len": predeclaredFunction,
|
|
||||||
"make": predeclaredFunction,
|
|
||||||
"new": predeclaredFunction,
|
|
||||||
"panic": predeclaredFunction,
|
|
||||||
"print": predeclaredFunction,
|
|
||||||
"println": predeclaredFunction,
|
|
||||||
"real": predeclaredFunction,
|
|
||||||
"recover": predeclaredFunction,
|
|
||||||
}
|
|
||||||
|
|
||||||
const (
|
|
||||||
ExportLinkAnnotation AnnotationKind = iota
|
|
||||||
AnchorAnnotation
|
|
||||||
CommentAnnotation
|
|
||||||
PackageLinkAnnotation
|
|
||||||
BuiltinAnnotation
|
|
||||||
)
|
|
||||||
|
|
||||||
// annotationVisitor collects annotations.
|
|
||||||
type annotationVisitor struct {
|
|
||||||
annotations []Annotation
|
|
||||||
}
|
|
||||||
|
|
||||||
func (v *annotationVisitor) add(kind AnnotationKind, importPath string) {
|
|
||||||
v.annotations = append(v.annotations, Annotation{Kind: kind, ImportPath: importPath})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (v *annotationVisitor) ignoreName() {
|
|
||||||
v.add(-1, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (v *annotationVisitor) Visit(n ast.Node) ast.Visitor {
|
|
||||||
switch n := n.(type) {
|
|
||||||
case *ast.TypeSpec:
|
|
||||||
v.ignoreName()
|
|
||||||
ast.Walk(v, n.Type)
|
|
||||||
case *ast.FuncDecl:
|
|
||||||
if n.Recv != nil {
|
|
||||||
ast.Walk(v, n.Recv)
|
|
||||||
}
|
|
||||||
v.ignoreName()
|
|
||||||
ast.Walk(v, n.Type)
|
|
||||||
case *ast.Field:
|
|
||||||
for range n.Names {
|
|
||||||
v.ignoreName()
|
|
||||||
}
|
|
||||||
ast.Walk(v, n.Type)
|
|
||||||
case *ast.ValueSpec:
|
|
||||||
for range n.Names {
|
|
||||||
v.add(AnchorAnnotation, "")
|
|
||||||
}
|
|
||||||
if n.Type != nil {
|
|
||||||
ast.Walk(v, n.Type)
|
|
||||||
}
|
|
||||||
for _, x := range n.Values {
|
|
||||||
ast.Walk(v, x)
|
|
||||||
}
|
|
||||||
case *ast.Ident:
|
|
||||||
switch {
|
|
||||||
case n.Obj == nil && predeclared[n.Name] != notPredeclared:
|
|
||||||
v.add(BuiltinAnnotation, "")
|
|
||||||
case n.Obj != nil && ast.IsExported(n.Name):
|
|
||||||
v.add(ExportLinkAnnotation, "")
|
|
||||||
default:
|
|
||||||
v.ignoreName()
|
|
||||||
}
|
|
||||||
case *ast.SelectorExpr:
|
|
||||||
if x, _ := n.X.(*ast.Ident); x != nil {
|
|
||||||
if obj := x.Obj; obj != nil && obj.Kind == ast.Pkg {
|
|
||||||
if spec, _ := obj.Decl.(*ast.ImportSpec); spec != nil {
|
|
||||||
if path, err := strconv.Unquote(spec.Path.Value); err == nil {
|
|
||||||
v.add(PackageLinkAnnotation, path)
|
|
||||||
if path == "C" {
|
|
||||||
v.ignoreName()
|
|
||||||
} else {
|
|
||||||
v.add(ExportLinkAnnotation, path)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
ast.Walk(v, n.X)
|
|
||||||
v.ignoreName()
|
|
||||||
default:
|
|
||||||
return v
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func printDecl(decl ast.Node, fset *token.FileSet, buf []byte) (Code, []byte) {
|
|
||||||
v := &annotationVisitor{}
|
|
||||||
ast.Walk(v, decl)
|
|
||||||
|
|
||||||
buf = buf[:0]
|
|
||||||
err := (&printer.Config{Mode: printer.UseSpaces, Tabwidth: 4}).Fprint(sliceWriter{&buf}, fset, decl)
|
|
||||||
if err != nil {
|
|
||||||
return Code{Text: err.Error()}, buf
|
|
||||||
}
|
|
||||||
|
|
||||||
var annotations []Annotation
|
|
||||||
var s scanner.Scanner
|
|
||||||
fset = token.NewFileSet()
|
|
||||||
file := fset.AddFile("", fset.Base(), len(buf))
|
|
||||||
s.Init(file, buf, nil, scanner.ScanComments)
|
|
||||||
loop:
|
|
||||||
for {
|
|
||||||
pos, tok, lit := s.Scan()
|
|
||||||
switch tok {
|
|
||||||
case token.EOF:
|
|
||||||
break loop
|
|
||||||
case token.COMMENT:
|
|
||||||
p := file.Offset(pos)
|
|
||||||
e := p + len(lit)
|
|
||||||
if p > math.MaxInt16 || e > math.MaxInt16 {
|
|
||||||
break loop
|
|
||||||
}
|
|
||||||
annotations = append(annotations, Annotation{Kind: CommentAnnotation, Pos: int16(p), End: int16(e)})
|
|
||||||
case token.IDENT:
|
|
||||||
if len(v.annotations) == 0 {
|
|
||||||
// Oops!
|
|
||||||
break loop
|
|
||||||
}
|
|
||||||
annotation := v.annotations[0]
|
|
||||||
v.annotations = v.annotations[1:]
|
|
||||||
if annotation.Kind == -1 {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
p := file.Offset(pos)
|
|
||||||
e := p + len(lit)
|
|
||||||
if p > math.MaxInt16 || e > math.MaxInt16 {
|
|
||||||
break loop
|
|
||||||
}
|
|
||||||
annotation.Pos = int16(p)
|
|
||||||
annotation.End = int16(e)
|
|
||||||
if len(annotations) > 0 && annotation.Kind == ExportLinkAnnotation {
|
|
||||||
prev := annotations[len(annotations)-1]
|
|
||||||
if prev.Kind == PackageLinkAnnotation &&
|
|
||||||
prev.ImportPath == annotation.ImportPath &&
|
|
||||||
prev.End+1 == annotation.Pos {
|
|
||||||
// merge with previous
|
|
||||||
annotation.Pos = prev.Pos
|
|
||||||
annotations[len(annotations)-1] = annotation
|
|
||||||
continue loop
|
|
||||||
}
|
|
||||||
}
|
|
||||||
annotations = append(annotations, annotation)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return Code{Text: string(buf), Annotations: annotations}, buf
|
|
||||||
}
|
|
||||||
|
|
||||||
type AnnotationKind int16
|
|
||||||
|
|
||||||
type Annotation struct {
|
|
||||||
Pos, End int16
|
|
||||||
Kind AnnotationKind
|
|
||||||
ImportPath string
|
|
||||||
}
|
|
||||||
|
|
||||||
type Code struct {
|
|
||||||
Text string
|
|
||||||
Annotations []Annotation
|
|
||||||
}
|
|
||||||
|
|
||||||
func commentAnnotations(src string) []Annotation {
|
|
||||||
var annotations []Annotation
|
|
||||||
var s scanner.Scanner
|
|
||||||
fset := token.NewFileSet()
|
|
||||||
file := fset.AddFile("", fset.Base(), len(src))
|
|
||||||
s.Init(file, []byte(src), nil, scanner.ScanComments)
|
|
||||||
for {
|
|
||||||
pos, tok, lit := s.Scan()
|
|
||||||
switch tok {
|
|
||||||
case token.EOF:
|
|
||||||
return annotations
|
|
||||||
case token.COMMENT:
|
|
||||||
p := file.Offset(pos)
|
|
||||||
e := p + len(lit)
|
|
||||||
if p > math.MaxInt16 || e > math.MaxInt16 {
|
|
||||||
return annotations
|
|
||||||
}
|
|
||||||
annotations = append(annotations, Annotation{Kind: CommentAnnotation, Pos: int16(p), End: int16(e)})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type sliceWriter struct{ p *[]byte }
|
|
||||||
|
|
||||||
func (w sliceWriter) Write(p []byte) (int, error) {
|
|
||||||
*w.p = append(*w.p, p...)
|
|
||||||
return len(p), nil
|
|
||||||
}
|
|
177
conf.go
177
conf.go
@ -1,177 +0,0 @@
|
|||||||
// Copyright 2013 bee 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 main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
"io/ioutil"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"io"
|
|
||||||
"path/filepath"
|
|
||||||
|
|
||||||
"gopkg.in/yaml.v2"
|
|
||||||
)
|
|
||||||
|
|
||||||
const confVer = 0
|
|
||||||
|
|
||||||
var defaultConf = `{
|
|
||||||
"version": 0,
|
|
||||||
"gopm": {
|
|
||||||
"enable": false,
|
|
||||||
"install": false
|
|
||||||
},
|
|
||||||
"go_install": true,
|
|
||||||
"watch_ext": [],
|
|
||||||
"dir_structure": {
|
|
||||||
"watch_all": false,
|
|
||||||
"controllers": "",
|
|
||||||
"models": "",
|
|
||||||
"others": []
|
|
||||||
},
|
|
||||||
"cmd_args": [],
|
|
||||||
"envs": [],
|
|
||||||
"database": {
|
|
||||||
"driver": "mysql"
|
|
||||||
},
|
|
||||||
"enable_reload": false
|
|
||||||
}
|
|
||||||
`
|
|
||||||
var conf struct {
|
|
||||||
Version int
|
|
||||||
// gopm support
|
|
||||||
Gopm struct {
|
|
||||||
Enable bool
|
|
||||||
Install bool
|
|
||||||
}
|
|
||||||
// Indicates whether execute "go install" before "go build".
|
|
||||||
GoInstall bool `json:"go_install" yaml:"go_install"`
|
|
||||||
WatchExt []string `json:"watch_ext" yaml:"watch_ext"`
|
|
||||||
DirStruct struct {
|
|
||||||
WatchAll bool `json:"watch_all" yaml:"watch_all"`
|
|
||||||
Controllers string
|
|
||||||
Models string
|
|
||||||
Others []string // Other directories.
|
|
||||||
} `json:"dir_structure" yaml:"dir_structure"`
|
|
||||||
CmdArgs []string `json:"cmd_args" yaml:"cmd_args"`
|
|
||||||
Envs []string
|
|
||||||
Bale struct {
|
|
||||||
Import string
|
|
||||||
Dirs []string
|
|
||||||
IngExt []string `json:"ignore_ext" yaml:"ignore_ext"`
|
|
||||||
}
|
|
||||||
Database struct {
|
|
||||||
Driver string
|
|
||||||
Conn string
|
|
||||||
}
|
|
||||||
EnableReload bool `json:"enable_reload" yaml:"enable_reload"`
|
|
||||||
}
|
|
||||||
|
|
||||||
// loadConfig loads customized configuration.
|
|
||||||
func loadConfig() (err error) {
|
|
||||||
err = filepath.Walk(".", func(path string, fileInfo os.FileInfo, err error) error {
|
|
||||||
if err != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if fileInfo.IsDir() {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if fileInfo.Name() == "bee.json" {
|
|
||||||
logger.Info("Loading configuration from 'bee.json'...")
|
|
||||||
err = parseJSON(path, &conf)
|
|
||||||
if err != nil {
|
|
||||||
logger.Errorf("Failed to parse JSON file: %s", err)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return io.EOF
|
|
||||||
}
|
|
||||||
|
|
||||||
if fileInfo.Name() == "Beefile" {
|
|
||||||
logger.Info("Loading configuration from 'Beefile'...")
|
|
||||||
err = parseYAML(path, &conf)
|
|
||||||
if err != nil {
|
|
||||||
logger.Errorf("Failed to parse YAML file: %s", err)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return io.EOF
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
})
|
|
||||||
|
|
||||||
// In case no configuration file found or an error different than io.EOF,
|
|
||||||
// fallback to default configuration
|
|
||||||
if err != io.EOF {
|
|
||||||
logger.Info("Loading default configuration...")
|
|
||||||
err = json.Unmarshal([]byte(defaultConf), &conf)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// No need to return io.EOF error
|
|
||||||
err = nil
|
|
||||||
|
|
||||||
// Check format version
|
|
||||||
if conf.Version != confVer {
|
|
||||||
logger.Warn("Your configuration file is outdated. Please do consider updating it.")
|
|
||||||
logger.Hint("Check the latest version of bee's configuration file.")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set variables
|
|
||||||
if len(conf.DirStruct.Controllers) == 0 {
|
|
||||||
conf.DirStruct.Controllers = "controllers"
|
|
||||||
}
|
|
||||||
if len(conf.DirStruct.Models) == 0 {
|
|
||||||
conf.DirStruct.Models = "models"
|
|
||||||
}
|
|
||||||
|
|
||||||
// Append watch exts
|
|
||||||
watchExts = append(watchExts, conf.WatchExt...)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func parseJSON(path string, v interface{}) error {
|
|
||||||
var (
|
|
||||||
data []byte
|
|
||||||
err error
|
|
||||||
)
|
|
||||||
data, err = ioutil.ReadFile(path)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
err = json.Unmarshal(data, v)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func parseYAML(path string, v interface{}) error {
|
|
||||||
var (
|
|
||||||
data []byte
|
|
||||||
err error
|
|
||||||
)
|
|
||||||
data, err = ioutil.ReadFile(path)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
err = yaml.Unmarshal(data, v)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
159
config/conf.go
Normal file
159
config/conf.go
Normal file
@ -0,0 +1,159 @@
|
|||||||
|
// Copyright 2013 bee 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 config
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"gopkg.in/yaml.v2"
|
||||||
|
)
|
||||||
|
|
||||||
|
const confVer = 0
|
||||||
|
|
||||||
|
var Conf = struct {
|
||||||
|
Version int
|
||||||
|
GoInstall bool `json:"go_install" yaml:"go_install"` // Indicates whether execute "go install" before "go build".
|
||||||
|
DirStruct dirStruct `json:"dir_structure" yaml:"dir_structure"`
|
||||||
|
CmdArgs []string `json:"cmd_args" yaml:"cmd_args"`
|
||||||
|
Envs []string
|
||||||
|
Bale bale
|
||||||
|
Database database
|
||||||
|
EnableReload bool `json:"enable_reload" yaml:"enable_reload"`
|
||||||
|
EnableNotification bool `json:"enable_notification" yaml:"enable_notification"`
|
||||||
|
Scripts map[string]string `json:"scripts" yaml:"scripts"`
|
||||||
|
}{
|
||||||
|
GoInstall: true,
|
||||||
|
DirStruct: dirStruct{
|
||||||
|
Others: []string{},
|
||||||
|
},
|
||||||
|
CmdArgs: []string{},
|
||||||
|
Envs: []string{},
|
||||||
|
Bale: bale{
|
||||||
|
Dirs: []string{},
|
||||||
|
IngExt: []string{},
|
||||||
|
},
|
||||||
|
Database: database{
|
||||||
|
Driver: "mysql",
|
||||||
|
},
|
||||||
|
EnableNotification: true,
|
||||||
|
Scripts: map[string]string{},
|
||||||
|
}
|
||||||
|
|
||||||
|
// dirStruct describes the application's directory structure
|
||||||
|
type dirStruct struct {
|
||||||
|
WatchAll bool `json:"watch_all" yaml:"watch_all"`
|
||||||
|
Controllers string
|
||||||
|
Models string
|
||||||
|
Others []string // Other directories
|
||||||
|
}
|
||||||
|
|
||||||
|
// bale
|
||||||
|
type bale struct {
|
||||||
|
Import string
|
||||||
|
Dirs []string
|
||||||
|
IngExt []string `json:"ignore_ext" yaml:"ignore_ext"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// database holds the database connection information
|
||||||
|
type database struct {
|
||||||
|
Driver string
|
||||||
|
Conn string
|
||||||
|
}
|
||||||
|
|
||||||
|
// LoadConfig loads the bee tool configuration.
|
||||||
|
// It looks for Beefile or bee.json in the current path,
|
||||||
|
// and falls back to default configuration in case not found.
|
||||||
|
func LoadConfig() {
|
||||||
|
currentPath, err := os.Getwd()
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Error(err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
dir, err := os.Open(currentPath)
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Error(err.Error())
|
||||||
|
}
|
||||||
|
defer dir.Close()
|
||||||
|
|
||||||
|
files, err := dir.Readdir(-1)
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Error(err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, file := range files {
|
||||||
|
switch file.Name() {
|
||||||
|
case "bee.json":
|
||||||
|
{
|
||||||
|
err = parseJSON(filepath.Join(currentPath, file.Name()), &Conf)
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Errorf("Failed to parse JSON file: %s", err)
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
case "Beefile":
|
||||||
|
{
|
||||||
|
err = parseYAML(filepath.Join(currentPath, file.Name()), &Conf)
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Errorf("Failed to parse YAML file: %s", err)
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check format version
|
||||||
|
if Conf.Version != confVer {
|
||||||
|
beeLogger.Log.Warn("Your configuration file is outdated. Please do consider updating it.")
|
||||||
|
beeLogger.Log.Hint("Check the latest version of bee's configuration file.")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set variables
|
||||||
|
if len(Conf.DirStruct.Controllers) == 0 {
|
||||||
|
Conf.DirStruct.Controllers = "controllers"
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(Conf.DirStruct.Models) == 0 {
|
||||||
|
Conf.DirStruct.Models = "models"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseJSON(path string, v interface{}) error {
|
||||||
|
var (
|
||||||
|
data []byte
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
data, err = ioutil.ReadFile(path)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
err = json.Unmarshal(data, v)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseYAML(path string, v interface{}) error {
|
||||||
|
var (
|
||||||
|
data []byte
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
data, err = ioutil.ReadFile(path)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
err = yaml.Unmarshal(data, v)
|
||||||
|
return err
|
||||||
|
}
|
200
g.go
200
g.go
@ -1,200 +0,0 @@
|
|||||||
// Copyright 2013 bee 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 main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"os"
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
var cmdGenerate = &Command{
|
|
||||||
UsageLine: "generate [command]",
|
|
||||||
Short: "Source code generator",
|
|
||||||
Long: `▶ {{"To scaffold out your entire application:"|bold}}
|
|
||||||
|
|
||||||
$ bee generate scaffold [scaffoldname] [-fields="title:string,body:text"] [-driver=mysql] [-conn="root:@tcp(127.0.0.1:3306)/test"]
|
|
||||||
|
|
||||||
▶ {{"To generate a Model based on fields:"|bold}}
|
|
||||||
|
|
||||||
$ bee generate model [modelname] [-fields="name:type"]
|
|
||||||
|
|
||||||
▶ {{"To generate a controller:"|bold}}
|
|
||||||
|
|
||||||
$ bee generate controller [controllerfile]
|
|
||||||
|
|
||||||
▶ {{"To generate a CRUD view:"|bold}}
|
|
||||||
|
|
||||||
$ bee generate view [viewpath]
|
|
||||||
|
|
||||||
▶ {{"To generate a migration file for making database schema updates:"|bold}}
|
|
||||||
|
|
||||||
$ bee generate migration [migrationfile] [-fields="name:type"]
|
|
||||||
|
|
||||||
▶ {{"To generate swagger doc file:"|bold}}
|
|
||||||
|
|
||||||
$ bee generate docs
|
|
||||||
|
|
||||||
▶ {{"To generate a test case:"|bold}}
|
|
||||||
|
|
||||||
$ bee generate test [routerfile]
|
|
||||||
|
|
||||||
▶ {{"To generate appcode based on an existing database:"|bold}}
|
|
||||||
|
|
||||||
$ bee generate appcode [-tables=""] [-driver=mysql] [-conn="root:@tcp(127.0.0.1:3306)/test"] [-level=3]
|
|
||||||
`,
|
|
||||||
PreRun: func(cmd *Command, args []string) { ShowShortVersionBanner() },
|
|
||||||
Run: generateCode,
|
|
||||||
}
|
|
||||||
|
|
||||||
var driver docValue
|
|
||||||
var conn docValue
|
|
||||||
var level docValue
|
|
||||||
var tables docValue
|
|
||||||
var fields docValue
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
cmdGenerate.Flag.Var(&tables, "tables", "List of table names separated by a comma.")
|
|
||||||
cmdGenerate.Flag.Var(&driver, "driver", "Database driver. Either mysql, postgres or sqlite.")
|
|
||||||
cmdGenerate.Flag.Var(&conn, "conn", "Connection string used by the driver to connect to a database instance.")
|
|
||||||
cmdGenerate.Flag.Var(&level, "level", "Either 1, 2 or 3. i.e. 1=models; 2=models and controllers; 3=models, controllers and routers.")
|
|
||||||
cmdGenerate.Flag.Var(&fields, "fields", "List of table fields.")
|
|
||||||
}
|
|
||||||
|
|
||||||
func generateCode(cmd *Command, args []string) int {
|
|
||||||
currpath, _ := os.Getwd()
|
|
||||||
if len(args) < 1 {
|
|
||||||
logger.Fatal("Command is missing")
|
|
||||||
}
|
|
||||||
|
|
||||||
gps := GetGOPATHs()
|
|
||||||
if len(gps) == 0 {
|
|
||||||
logger.Fatal("GOPATH environment variable is not set or empty")
|
|
||||||
}
|
|
||||||
|
|
||||||
gopath := gps[0]
|
|
||||||
|
|
||||||
logger.Debugf("GOPATH: %s", __FILE__(), __LINE__(), gopath)
|
|
||||||
|
|
||||||
gcmd := args[0]
|
|
||||||
switch gcmd {
|
|
||||||
case "scaffold":
|
|
||||||
if len(args) < 2 {
|
|
||||||
logger.Fatal("Wrong number of arguments. Run: bee help generate")
|
|
||||||
}
|
|
||||||
// Load the configuration
|
|
||||||
err := loadConfig()
|
|
||||||
if err != nil {
|
|
||||||
logger.Fatalf("Failed to load configuration: %s", err)
|
|
||||||
}
|
|
||||||
cmd.Flag.Parse(args[2:])
|
|
||||||
if driver == "" {
|
|
||||||
driver = docValue(conf.Database.Driver)
|
|
||||||
if driver == "" {
|
|
||||||
driver = "mysql"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if conn == "" {
|
|
||||||
conn = docValue(conf.Database.Conn)
|
|
||||||
if conn == "" {
|
|
||||||
conn = "root:@tcp(127.0.0.1:3306)/test"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if fields == "" {
|
|
||||||
logger.Hint("fields option should not be empty, i.e. -fields=\"title:string,body:text\"")
|
|
||||||
logger.Fatal("Wrong number of arguments. Run: bee help generate")
|
|
||||||
}
|
|
||||||
sname := args[1]
|
|
||||||
generateScaffold(sname, fields.String(), currpath, driver.String(), conn.String())
|
|
||||||
case "docs":
|
|
||||||
generateDocs(currpath)
|
|
||||||
case "appcode":
|
|
||||||
// Load the configuration
|
|
||||||
err := loadConfig()
|
|
||||||
if err != nil {
|
|
||||||
logger.Fatalf("Failed to load configuration: %s", err)
|
|
||||||
}
|
|
||||||
cmd.Flag.Parse(args[1:])
|
|
||||||
if driver == "" {
|
|
||||||
driver = docValue(conf.Database.Driver)
|
|
||||||
if driver == "" {
|
|
||||||
driver = "mysql"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if conn == "" {
|
|
||||||
conn = docValue(conf.Database.Conn)
|
|
||||||
if conn == "" {
|
|
||||||
if driver == "mysql" {
|
|
||||||
conn = "root:@tcp(127.0.0.1:3306)/test"
|
|
||||||
} else if driver == "postgres" {
|
|
||||||
conn = "postgres://postgres:postgres@127.0.0.1:5432/postgres"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if level == "" {
|
|
||||||
level = "3"
|
|
||||||
}
|
|
||||||
logger.Infof("Using '%s' as 'driver'", driver)
|
|
||||||
logger.Infof("Using '%s' as 'conn'", conn)
|
|
||||||
logger.Infof("Using '%s' as 'tables'", tables)
|
|
||||||
logger.Infof("Using '%s' as 'level'", level)
|
|
||||||
generateAppcode(driver.String(), conn.String(), level.String(), tables.String(), currpath)
|
|
||||||
case "migration":
|
|
||||||
if len(args) < 2 {
|
|
||||||
logger.Fatal("Wrong number of arguments. Run: bee help generate")
|
|
||||||
}
|
|
||||||
cmd.Flag.Parse(args[2:])
|
|
||||||
mname := args[1]
|
|
||||||
|
|
||||||
logger.Infof("Using '%s' as migration name", mname)
|
|
||||||
|
|
||||||
upsql := ""
|
|
||||||
downsql := ""
|
|
||||||
if fields != "" {
|
|
||||||
dbMigrator := newDBDriver()
|
|
||||||
upsql = dbMigrator.generateCreateUp(mname)
|
|
||||||
downsql = dbMigrator.generateCreateDown(mname)
|
|
||||||
}
|
|
||||||
generateMigration(mname, upsql, downsql, currpath)
|
|
||||||
case "controller":
|
|
||||||
if len(args) == 2 {
|
|
||||||
cname := args[1]
|
|
||||||
generateController(cname, currpath)
|
|
||||||
} else {
|
|
||||||
logger.Fatal("Wrong number of arguments. Run: bee help generate")
|
|
||||||
}
|
|
||||||
case "model":
|
|
||||||
if len(args) < 2 {
|
|
||||||
logger.Fatal("Wrong number of arguments. Run: bee help generate")
|
|
||||||
}
|
|
||||||
cmd.Flag.Parse(args[2:])
|
|
||||||
if fields == "" {
|
|
||||||
logger.Hint("fields option should not be empty, i.e. -fields=\"title:string,body:text\"")
|
|
||||||
logger.Fatal("Wrong number of arguments. Run: bee help generate")
|
|
||||||
}
|
|
||||||
sname := args[1]
|
|
||||||
generateModel(sname, fields.String(), currpath)
|
|
||||||
case "view":
|
|
||||||
if len(args) == 2 {
|
|
||||||
cname := args[1]
|
|
||||||
generateView(cname, currpath)
|
|
||||||
} else {
|
|
||||||
logger.Fatal("Wrong number of arguments. Run: bee help generate")
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
logger.Fatal("Command is missing")
|
|
||||||
}
|
|
||||||
logger.Successf("%s successfully generated!", strings.Title(gcmd))
|
|
||||||
return 0
|
|
||||||
}
|
|
@ -1,48 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import "strings"
|
|
||||||
|
|
||||||
func generateScaffold(sname, fields, currpath, driver, conn string) {
|
|
||||||
logger.Infof("Do you want to create a '%s' model? [Yes|No] ", sname)
|
|
||||||
|
|
||||||
// Generate the model
|
|
||||||
if askForConfirmation() {
|
|
||||||
generateModel(sname, fields, currpath)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Generate the controller
|
|
||||||
logger.Infof("Do you want to create a '%s' controller? [Yes|No] ", sname)
|
|
||||||
if askForConfirmation() {
|
|
||||||
generateController(sname, currpath)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Generate the views
|
|
||||||
logger.Infof("Do you want to create views for this '%s' resource? [Yes|No] ", sname)
|
|
||||||
if askForConfirmation() {
|
|
||||||
generateView(sname, currpath)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Generate a migration
|
|
||||||
logger.Infof("Do you want to create a '%s' migration and schema for this resource? [Yes|No] ", sname)
|
|
||||||
if askForConfirmation() {
|
|
||||||
upsql := ""
|
|
||||||
downsql := ""
|
|
||||||
if fields != "" {
|
|
||||||
dbMigrator := newDBDriver()
|
|
||||||
upsql = dbMigrator.generateCreateUp(sname)
|
|
||||||
downsql = dbMigrator.generateCreateDown(sname)
|
|
||||||
//todo remove
|
|
||||||
//if driver == "" {
|
|
||||||
// downsql = strings.Replace(downsql, "`", "", -1)
|
|
||||||
//}
|
|
||||||
}
|
|
||||||
generateMigration(sname, upsql, downsql, currpath)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Run the migration
|
|
||||||
logger.Infof("Do you want to migrate the database? [Yes|No] ")
|
|
||||||
if askForConfirmation() {
|
|
||||||
migrateUpdate(currpath, driver, conn)
|
|
||||||
}
|
|
||||||
logger.Successf("All done! Don't forget to add beego.Router(\"/%s\" ,&controllers.%sController{}) to routers/route.go\n", sname, strings.Title(sname))
|
|
||||||
}
|
|
23
generate/g.go
Normal file
23
generate/g.go
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
// Copyright 2013 bee 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 generate
|
||||||
|
|
||||||
|
import "github.com/beego/bee/utils"
|
||||||
|
|
||||||
|
var SQLDriver utils.DocValue
|
||||||
|
var SQLConn utils.DocValue
|
||||||
|
var Level utils.DocValue
|
||||||
|
var Tables utils.DocValue
|
||||||
|
var Fields utils.DocValue
|
@ -12,7 +12,7 @@
|
|||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
|
||||||
package main
|
package generate
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"database/sql"
|
"database/sql"
|
||||||
@ -23,6 +23,9 @@ import (
|
|||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/logger/colors"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
_ "github.com/go-sql-driver/mysql"
|
_ "github.com/go-sql-driver/mysql"
|
||||||
_ "github.com/lib/pq"
|
_ "github.com/lib/pq"
|
||||||
)
|
)
|
||||||
@ -183,7 +186,7 @@ type OrmTag struct {
|
|||||||
|
|
||||||
// String returns the source code string for the Table struct
|
// String returns the source code string for the Table struct
|
||||||
func (tb *Table) String() string {
|
func (tb *Table) String() string {
|
||||||
rv := fmt.Sprintf("type %s struct {\n", camelCase(tb.Name))
|
rv := fmt.Sprintf("type %s struct {\n", utils.CamelCase(tb.Name))
|
||||||
for _, v := range tb.Columns {
|
for _, v := range tb.Columns {
|
||||||
rv += v.String() + "\n"
|
rv += v.String() + "\n"
|
||||||
}
|
}
|
||||||
@ -255,7 +258,7 @@ func (tag *OrmTag) String() string {
|
|||||||
return fmt.Sprintf("`orm:\"%s\"`", strings.Join(ormOptions, ";"))
|
return fmt.Sprintf("`orm:\"%s\"`", strings.Join(ormOptions, ";"))
|
||||||
}
|
}
|
||||||
|
|
||||||
func generateAppcode(driver, connStr, level, tables, currpath string) {
|
func GenerateAppcode(driver, connStr, level, tables, currpath string) {
|
||||||
var mode byte
|
var mode byte
|
||||||
switch level {
|
switch level {
|
||||||
case "1":
|
case "1":
|
||||||
@ -265,7 +268,7 @@ func generateAppcode(driver, connStr, level, tables, currpath string) {
|
|||||||
case "3":
|
case "3":
|
||||||
mode = OModel | OController | ORouter
|
mode = OModel | OController | ORouter
|
||||||
default:
|
default:
|
||||||
logger.Fatal("Invalid level value. Must be either \"1\", \"2\", or \"3\"")
|
beeLogger.Log.Fatal("Invalid level value. Must be either \"1\", \"2\", or \"3\"")
|
||||||
}
|
}
|
||||||
var selectedTables map[string]bool
|
var selectedTables map[string]bool
|
||||||
if tables != "" {
|
if tables != "" {
|
||||||
@ -278,9 +281,9 @@ func generateAppcode(driver, connStr, level, tables, currpath string) {
|
|||||||
case "mysql":
|
case "mysql":
|
||||||
case "postgres":
|
case "postgres":
|
||||||
case "sqlite":
|
case "sqlite":
|
||||||
logger.Fatal("Generating app code from SQLite database is not supported yet.")
|
beeLogger.Log.Fatal("Generating app code from SQLite database is not supported yet.")
|
||||||
default:
|
default:
|
||||||
logger.Fatal("Unknown database driver. Must be either \"mysql\", \"postgres\" or \"sqlite\"")
|
beeLogger.Log.Fatal("Unknown database driver. Must be either \"mysql\", \"postgres\" or \"sqlite\"")
|
||||||
}
|
}
|
||||||
gen(driver, connStr, mode, selectedTables, currpath)
|
gen(driver, connStr, mode, selectedTables, currpath)
|
||||||
}
|
}
|
||||||
@ -290,11 +293,11 @@ func generateAppcode(driver, connStr, level, tables, currpath string) {
|
|||||||
func gen(dbms, connStr string, mode byte, selectedTableNames map[string]bool, apppath string) {
|
func gen(dbms, connStr string, mode byte, selectedTableNames map[string]bool, apppath string) {
|
||||||
db, err := sql.Open(dbms, connStr)
|
db, err := sql.Open(dbms, connStr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Could not connect to '%s' database using '%s': %s", dbms, connStr, err)
|
beeLogger.Log.Fatalf("Could not connect to '%s' database using '%s': %s", dbms, connStr, err)
|
||||||
}
|
}
|
||||||
defer db.Close()
|
defer db.Close()
|
||||||
if trans, ok := dbDriver[dbms]; ok {
|
if trans, ok := dbDriver[dbms]; ok {
|
||||||
logger.Info("Analyzing database tables...")
|
beeLogger.Log.Info("Analyzing database tables...")
|
||||||
tableNames := trans.GetTableNames(db)
|
tableNames := trans.GetTableNames(db)
|
||||||
tables := getTableObjects(tableNames, db, trans)
|
tables := getTableObjects(tableNames, db, trans)
|
||||||
mvcPath := new(MvcPath)
|
mvcPath := new(MvcPath)
|
||||||
@ -305,7 +308,7 @@ func gen(dbms, connStr string, mode byte, selectedTableNames map[string]bool, ap
|
|||||||
pkgPath := getPackagePath(apppath)
|
pkgPath := getPackagePath(apppath)
|
||||||
writeSourceFiles(pkgPath, tables, mode, mvcPath, selectedTableNames)
|
writeSourceFiles(pkgPath, tables, mode, mvcPath, selectedTableNames)
|
||||||
} else {
|
} else {
|
||||||
logger.Fatalf("Generating app code from '%s' database is not supported yet.", dbms)
|
beeLogger.Log.Fatalf("Generating app code from '%s' database is not supported yet.", dbms)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -313,13 +316,13 @@ func gen(dbms, connStr string, mode byte, selectedTableNames map[string]bool, ap
|
|||||||
func (*MysqlDB) GetTableNames(db *sql.DB) (tables []string) {
|
func (*MysqlDB) GetTableNames(db *sql.DB) (tables []string) {
|
||||||
rows, err := db.Query("SHOW TABLES")
|
rows, err := db.Query("SHOW TABLES")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Could not show tables: %s", err)
|
beeLogger.Log.Fatalf("Could not show tables: %s", err)
|
||||||
}
|
}
|
||||||
defer rows.Close()
|
defer rows.Close()
|
||||||
for rows.Next() {
|
for rows.Next() {
|
||||||
var name string
|
var name string
|
||||||
if err := rows.Scan(&name); err != nil {
|
if err := rows.Scan(&name); err != nil {
|
||||||
logger.Fatalf("Could not show tables: %s", err)
|
beeLogger.Log.Fatalf("Could not show tables: %s", err)
|
||||||
}
|
}
|
||||||
tables = append(tables, name)
|
tables = append(tables, name)
|
||||||
}
|
}
|
||||||
@ -362,12 +365,12 @@ func (*MysqlDB) GetConstraints(db *sql.DB, table *Table, blackList map[string]bo
|
|||||||
c.table_schema = database() AND c.table_name = ? AND u.table_schema = database() AND u.table_name = ?`,
|
c.table_schema = database() AND c.table_name = ? AND u.table_schema = database() AND u.table_name = ?`,
|
||||||
table.Name, table.Name) // u.position_in_unique_constraint,
|
table.Name, table.Name) // u.position_in_unique_constraint,
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatal("Could not query INFORMATION_SCHEMA for PK/UK/FK information")
|
beeLogger.Log.Fatal("Could not query INFORMATION_SCHEMA for PK/UK/FK information")
|
||||||
}
|
}
|
||||||
for rows.Next() {
|
for rows.Next() {
|
||||||
var constraintTypeBytes, columnNameBytes, refTableSchemaBytes, refTableNameBytes, refColumnNameBytes, refOrdinalPosBytes []byte
|
var constraintTypeBytes, columnNameBytes, refTableSchemaBytes, refTableNameBytes, refColumnNameBytes, refOrdinalPosBytes []byte
|
||||||
if err := rows.Scan(&constraintTypeBytes, &columnNameBytes, &refTableSchemaBytes, &refTableNameBytes, &refColumnNameBytes, &refOrdinalPosBytes); err != nil {
|
if err := rows.Scan(&constraintTypeBytes, &columnNameBytes, &refTableSchemaBytes, &refTableNameBytes, &refColumnNameBytes, &refOrdinalPosBytes); err != nil {
|
||||||
logger.Fatal("Could not read INFORMATION_SCHEMA for PK/UK/FK information")
|
beeLogger.Log.Fatal("Could not read INFORMATION_SCHEMA for PK/UK/FK information")
|
||||||
}
|
}
|
||||||
constraintType, columnName, refTableSchema, refTableName, refColumnName, refOrdinalPos :=
|
constraintType, columnName, refTableSchema, refTableName, refColumnName, refOrdinalPos :=
|
||||||
string(constraintTypeBytes), string(columnNameBytes), string(refTableSchemaBytes),
|
string(constraintTypeBytes), string(columnNameBytes), string(refTableSchemaBytes),
|
||||||
@ -407,7 +410,7 @@ func (mysqlDB *MysqlDB) GetColumns(db *sql.DB, table *Table, blackList map[strin
|
|||||||
table_schema = database() AND table_name = ?`,
|
table_schema = database() AND table_name = ?`,
|
||||||
table.Name)
|
table.Name)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Could not query the database: %s", err)
|
beeLogger.Log.Fatalf("Could not query the database: %s", err)
|
||||||
}
|
}
|
||||||
defer colDefRows.Close()
|
defer colDefRows.Close()
|
||||||
|
|
||||||
@ -415,17 +418,17 @@ func (mysqlDB *MysqlDB) GetColumns(db *sql.DB, table *Table, blackList map[strin
|
|||||||
// datatype as bytes so that SQL <null> values can be retrieved
|
// datatype as bytes so that SQL <null> values can be retrieved
|
||||||
var colNameBytes, dataTypeBytes, columnTypeBytes, isNullableBytes, columnDefaultBytes, extraBytes []byte
|
var colNameBytes, dataTypeBytes, columnTypeBytes, isNullableBytes, columnDefaultBytes, extraBytes []byte
|
||||||
if err := colDefRows.Scan(&colNameBytes, &dataTypeBytes, &columnTypeBytes, &isNullableBytes, &columnDefaultBytes, &extraBytes); err != nil {
|
if err := colDefRows.Scan(&colNameBytes, &dataTypeBytes, &columnTypeBytes, &isNullableBytes, &columnDefaultBytes, &extraBytes); err != nil {
|
||||||
logger.Fatal("Could not query INFORMATION_SCHEMA for column information")
|
beeLogger.Log.Fatal("Could not query INFORMATION_SCHEMA for column information")
|
||||||
}
|
}
|
||||||
colName, dataType, columnType, isNullable, columnDefault, extra :=
|
colName, dataType, columnType, isNullable, columnDefault, extra :=
|
||||||
string(colNameBytes), string(dataTypeBytes), string(columnTypeBytes), string(isNullableBytes), string(columnDefaultBytes), string(extraBytes)
|
string(colNameBytes), string(dataTypeBytes), string(columnTypeBytes), string(isNullableBytes), string(columnDefaultBytes), string(extraBytes)
|
||||||
|
|
||||||
// create a column
|
// create a column
|
||||||
col := new(Column)
|
col := new(Column)
|
||||||
col.Name = camelCase(colName)
|
col.Name = utils.CamelCase(colName)
|
||||||
col.Type, err = mysqlDB.GetGoDataType(dataType)
|
col.Type, err = mysqlDB.GetGoDataType(dataType)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("%s", err)
|
beeLogger.Log.Fatalf("%s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Tag info
|
// Tag info
|
||||||
@ -449,8 +452,8 @@ func (mysqlDB *MysqlDB) GetColumns(db *sql.DB, table *Table, blackList map[strin
|
|||||||
if isFk && !isBl {
|
if isFk && !isBl {
|
||||||
tag.RelFk = true
|
tag.RelFk = true
|
||||||
refStructName := fkCol.RefTable
|
refStructName := fkCol.RefTable
|
||||||
col.Name = camelCase(colName)
|
col.Name = utils.CamelCase(colName)
|
||||||
col.Type = "*" + camelCase(refStructName)
|
col.Type = "*" + utils.CamelCase(refStructName)
|
||||||
} else {
|
} else {
|
||||||
// if the name of column is Id, and it's not primary key
|
// if the name of column is Id, and it's not primary key
|
||||||
if colName == "id" {
|
if colName == "id" {
|
||||||
@ -464,7 +467,7 @@ func (mysqlDB *MysqlDB) GetColumns(db *sql.DB, table *Table, blackList map[strin
|
|||||||
if sign == "unsigned" && extra != "auto_increment" {
|
if sign == "unsigned" && extra != "auto_increment" {
|
||||||
col.Type, err = mysqlDB.GetGoDataType(dataType + " " + sign)
|
col.Type, err = mysqlDB.GetGoDataType(dataType + " " + sign)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("%s", err)
|
beeLogger.Log.Fatalf("%s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -516,14 +519,14 @@ func (*PostgresDB) GetTableNames(db *sql.DB) (tables []string) {
|
|||||||
table_type = 'BASE TABLE' AND
|
table_type = 'BASE TABLE' AND
|
||||||
table_schema NOT IN ('pg_catalog', 'information_schema')`)
|
table_schema NOT IN ('pg_catalog', 'information_schema')`)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Could not show tables: %s", err)
|
beeLogger.Log.Fatalf("Could not show tables: %s", err)
|
||||||
}
|
}
|
||||||
defer rows.Close()
|
defer rows.Close()
|
||||||
|
|
||||||
for rows.Next() {
|
for rows.Next() {
|
||||||
var name string
|
var name string
|
||||||
if err := rows.Scan(&name); err != nil {
|
if err := rows.Scan(&name); err != nil {
|
||||||
logger.Fatalf("Could not show tables: %s", err)
|
beeLogger.Log.Fatalf("Could not show tables: %s", err)
|
||||||
}
|
}
|
||||||
tables = append(tables, name)
|
tables = append(tables, name)
|
||||||
}
|
}
|
||||||
@ -553,13 +556,13 @@ func (*PostgresDB) GetConstraints(db *sql.DB, table *Table, blackList map[string
|
|||||||
AND u.table_name = $2`,
|
AND u.table_name = $2`,
|
||||||
table.Name, table.Name) // u.position_in_unique_constraint,
|
table.Name, table.Name) // u.position_in_unique_constraint,
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Could not query INFORMATION_SCHEMA for PK/UK/FK information: %s", err)
|
beeLogger.Log.Fatalf("Could not query INFORMATION_SCHEMA for PK/UK/FK information: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
for rows.Next() {
|
for rows.Next() {
|
||||||
var constraintTypeBytes, columnNameBytes, refTableSchemaBytes, refTableNameBytes, refColumnNameBytes, refOrdinalPosBytes []byte
|
var constraintTypeBytes, columnNameBytes, refTableSchemaBytes, refTableNameBytes, refColumnNameBytes, refOrdinalPosBytes []byte
|
||||||
if err := rows.Scan(&constraintTypeBytes, &columnNameBytes, &refTableSchemaBytes, &refTableNameBytes, &refColumnNameBytes, &refOrdinalPosBytes); err != nil {
|
if err := rows.Scan(&constraintTypeBytes, &columnNameBytes, &refTableSchemaBytes, &refTableNameBytes, &refColumnNameBytes, &refOrdinalPosBytes); err != nil {
|
||||||
logger.Fatalf("Could not read INFORMATION_SCHEMA for PK/UK/FK information: %s", err)
|
beeLogger.Log.Fatalf("Could not read INFORMATION_SCHEMA for PK/UK/FK information: %s", err)
|
||||||
}
|
}
|
||||||
constraintType, columnName, refTableSchema, refTableName, refColumnName, refOrdinalPos :=
|
constraintType, columnName, refTableSchema, refTableName, refColumnName, refOrdinalPos :=
|
||||||
string(constraintTypeBytes), string(columnNameBytes), string(refTableSchemaBytes),
|
string(constraintTypeBytes), string(columnNameBytes), string(refTableSchemaBytes),
|
||||||
@ -609,7 +612,7 @@ func (postgresDB *PostgresDB) GetColumns(db *sql.DB, table *Table, blackList map
|
|||||||
AND table_name = $1`,
|
AND table_name = $1`,
|
||||||
table.Name)
|
table.Name)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Could not query INFORMATION_SCHEMA for column information: %s", err)
|
beeLogger.Log.Fatalf("Could not query INFORMATION_SCHEMA for column information: %s", err)
|
||||||
}
|
}
|
||||||
defer colDefRows.Close()
|
defer colDefRows.Close()
|
||||||
|
|
||||||
@ -617,16 +620,16 @@ func (postgresDB *PostgresDB) GetColumns(db *sql.DB, table *Table, blackList map
|
|||||||
// datatype as bytes so that SQL <null> values can be retrieved
|
// datatype as bytes so that SQL <null> values can be retrieved
|
||||||
var colNameBytes, dataTypeBytes, columnTypeBytes, isNullableBytes, columnDefaultBytes, extraBytes []byte
|
var colNameBytes, dataTypeBytes, columnTypeBytes, isNullableBytes, columnDefaultBytes, extraBytes []byte
|
||||||
if err := colDefRows.Scan(&colNameBytes, &dataTypeBytes, &columnTypeBytes, &isNullableBytes, &columnDefaultBytes, &extraBytes); err != nil {
|
if err := colDefRows.Scan(&colNameBytes, &dataTypeBytes, &columnTypeBytes, &isNullableBytes, &columnDefaultBytes, &extraBytes); err != nil {
|
||||||
logger.Fatalf("Could not query INFORMATION_SCHEMA for column information: %s", err)
|
beeLogger.Log.Fatalf("Could not query INFORMATION_SCHEMA for column information: %s", err)
|
||||||
}
|
}
|
||||||
colName, dataType, columnType, isNullable, columnDefault, extra :=
|
colName, dataType, columnType, isNullable, columnDefault, extra :=
|
||||||
string(colNameBytes), string(dataTypeBytes), string(columnTypeBytes), string(isNullableBytes), string(columnDefaultBytes), string(extraBytes)
|
string(colNameBytes), string(dataTypeBytes), string(columnTypeBytes), string(isNullableBytes), string(columnDefaultBytes), string(extraBytes)
|
||||||
// Create a column
|
// Create a column
|
||||||
col := new(Column)
|
col := new(Column)
|
||||||
col.Name = camelCase(colName)
|
col.Name = utils.CamelCase(colName)
|
||||||
col.Type, err = postgresDB.GetGoDataType(dataType)
|
col.Type, err = postgresDB.GetGoDataType(dataType)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("%s", err)
|
beeLogger.Log.Fatalf("%s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Tag info
|
// Tag info
|
||||||
@ -650,8 +653,8 @@ func (postgresDB *PostgresDB) GetColumns(db *sql.DB, table *Table, blackList map
|
|||||||
if isFk && !isBl {
|
if isFk && !isBl {
|
||||||
tag.RelFk = true
|
tag.RelFk = true
|
||||||
refStructName := fkCol.RefTable
|
refStructName := fkCol.RefTable
|
||||||
col.Name = camelCase(colName)
|
col.Name = utils.CamelCase(colName)
|
||||||
col.Type = "*" + camelCase(refStructName)
|
col.Type = "*" + utils.CamelCase(refStructName)
|
||||||
} else {
|
} else {
|
||||||
// if the name of column is Id, and it's not primary key
|
// if the name of column is Id, and it's not primary key
|
||||||
if colName == "id" {
|
if colName == "id" {
|
||||||
@ -716,22 +719,22 @@ func createPaths(mode byte, paths *MvcPath) {
|
|||||||
// Newly geneated files will be inside these folders.
|
// Newly geneated files will be inside these folders.
|
||||||
func writeSourceFiles(pkgPath string, tables []*Table, mode byte, paths *MvcPath, selectedTables map[string]bool) {
|
func writeSourceFiles(pkgPath string, tables []*Table, mode byte, paths *MvcPath, selectedTables map[string]bool) {
|
||||||
if (OModel & mode) == OModel {
|
if (OModel & mode) == OModel {
|
||||||
logger.Info("Creating model files...")
|
beeLogger.Log.Info("Creating model files...")
|
||||||
writeModelFiles(tables, paths.ModelPath, selectedTables)
|
writeModelFiles(tables, paths.ModelPath, selectedTables)
|
||||||
}
|
}
|
||||||
if (OController & mode) == OController {
|
if (OController & mode) == OController {
|
||||||
logger.Info("Creating controller files...")
|
beeLogger.Log.Info("Creating controller files...")
|
||||||
writeControllerFiles(tables, paths.ControllerPath, selectedTables, pkgPath)
|
writeControllerFiles(tables, paths.ControllerPath, selectedTables, pkgPath)
|
||||||
}
|
}
|
||||||
if (ORouter & mode) == ORouter {
|
if (ORouter & mode) == ORouter {
|
||||||
logger.Info("Creating router files...")
|
beeLogger.Log.Info("Creating router files...")
|
||||||
writeRouterFile(tables, paths.RouterPath, selectedTables, pkgPath)
|
writeRouterFile(tables, paths.RouterPath, selectedTables, pkgPath)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// writeModelFiles generates model files
|
// writeModelFiles generates model files
|
||||||
func writeModelFiles(tables []*Table, mPath string, selectedTables map[string]bool) {
|
func writeModelFiles(tables []*Table, mPath string, selectedTables map[string]bool) {
|
||||||
w := NewColorWriter(os.Stdout)
|
w := colors.NewColorWriter(os.Stdout)
|
||||||
|
|
||||||
for _, tb := range tables {
|
for _, tb := range tables {
|
||||||
// if selectedTables map is not nil and this table is not selected, ignore it
|
// if selectedTables map is not nil and this table is not selected, ignore it
|
||||||
@ -744,22 +747,22 @@ func writeModelFiles(tables []*Table, mPath string, selectedTables map[string]bo
|
|||||||
fpath := path.Join(mPath, filename+".go")
|
fpath := path.Join(mPath, filename+".go")
|
||||||
var f *os.File
|
var f *os.File
|
||||||
var err error
|
var err error
|
||||||
if isExist(fpath) {
|
if utils.IsExist(fpath) {
|
||||||
logger.Warnf("'%s' already exists. Do you want to overwrite it? [Yes|No] ", fpath)
|
beeLogger.Log.Warnf("'%s' already exists. Do you want to overwrite it? [Yes|No] ", fpath)
|
||||||
if askForConfirmation() {
|
if utils.AskForConfirmation() {
|
||||||
f, err = os.OpenFile(fpath, os.O_RDWR|os.O_TRUNC, 0666)
|
f, err = os.OpenFile(fpath, os.O_RDWR|os.O_TRUNC, 0666)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Warnf("%s", err)
|
beeLogger.Log.Warnf("%s", err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
logger.Warnf("Skipped create file '%s'", fpath)
|
beeLogger.Log.Warnf("Skipped create file '%s'", fpath)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
f, err = os.OpenFile(fpath, os.O_CREATE|os.O_RDWR, 0666)
|
f, err = os.OpenFile(fpath, os.O_CREATE|os.O_RDWR, 0666)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Warnf("%s", err)
|
beeLogger.Log.Warnf("%s", err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -770,7 +773,7 @@ func writeModelFiles(tables []*Table, mPath string, selectedTables map[string]bo
|
|||||||
template = ModelTPL
|
template = ModelTPL
|
||||||
}
|
}
|
||||||
fileStr := strings.Replace(template, "{{modelStruct}}", tb.String(), 1)
|
fileStr := strings.Replace(template, "{{modelStruct}}", tb.String(), 1)
|
||||||
fileStr = strings.Replace(fileStr, "{{modelName}}", camelCase(tb.Name), -1)
|
fileStr = strings.Replace(fileStr, "{{modelName}}", utils.CamelCase(tb.Name), -1)
|
||||||
fileStr = strings.Replace(fileStr, "{{tableName}}", tb.Name, -1)
|
fileStr = strings.Replace(fileStr, "{{tableName}}", tb.Name, -1)
|
||||||
|
|
||||||
// If table contains time field, import time.Time package
|
// If table contains time field, import time.Time package
|
||||||
@ -783,17 +786,17 @@ func writeModelFiles(tables []*Table, mPath string, selectedTables map[string]bo
|
|||||||
fileStr = strings.Replace(fileStr, "{{timePkg}}", timePkg, -1)
|
fileStr = strings.Replace(fileStr, "{{timePkg}}", timePkg, -1)
|
||||||
fileStr = strings.Replace(fileStr, "{{importTimePkg}}", importTimePkg, -1)
|
fileStr = strings.Replace(fileStr, "{{importTimePkg}}", importTimePkg, -1)
|
||||||
if _, err := f.WriteString(fileStr); err != nil {
|
if _, err := f.WriteString(fileStr); err != nil {
|
||||||
logger.Fatalf("Could not write model file to '%s': %s", fpath, err)
|
beeLogger.Log.Fatalf("Could not write model file to '%s': %s", fpath, err)
|
||||||
}
|
}
|
||||||
CloseFile(f)
|
utils.CloseFile(f)
|
||||||
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", fpath, "\x1b[0m")
|
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", fpath, "\x1b[0m")
|
||||||
formatSourceCode(fpath)
|
utils.FormatSourceCode(fpath)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// writeControllerFiles generates controller files
|
// writeControllerFiles generates controller files
|
||||||
func writeControllerFiles(tables []*Table, cPath string, selectedTables map[string]bool, pkgPath string) {
|
func writeControllerFiles(tables []*Table, cPath string, selectedTables map[string]bool, pkgPath string) {
|
||||||
w := NewColorWriter(os.Stdout)
|
w := colors.NewColorWriter(os.Stdout)
|
||||||
|
|
||||||
for _, tb := range tables {
|
for _, tb := range tables {
|
||||||
// If selectedTables map is not nil and this table is not selected, ignore it
|
// If selectedTables map is not nil and this table is not selected, ignore it
|
||||||
@ -809,39 +812,39 @@ func writeControllerFiles(tables []*Table, cPath string, selectedTables map[stri
|
|||||||
fpath := path.Join(cPath, filename+".go")
|
fpath := path.Join(cPath, filename+".go")
|
||||||
var f *os.File
|
var f *os.File
|
||||||
var err error
|
var err error
|
||||||
if isExist(fpath) {
|
if utils.IsExist(fpath) {
|
||||||
logger.Warnf("'%s' already exists. Do you want to overwrite it? [Yes|No] ", fpath)
|
beeLogger.Log.Warnf("'%s' already exists. Do you want to overwrite it? [Yes|No] ", fpath)
|
||||||
if askForConfirmation() {
|
if utils.AskForConfirmation() {
|
||||||
f, err = os.OpenFile(fpath, os.O_RDWR|os.O_TRUNC, 0666)
|
f, err = os.OpenFile(fpath, os.O_RDWR|os.O_TRUNC, 0666)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Warnf("%s", err)
|
beeLogger.Log.Warnf("%s", err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
logger.Warnf("Skipped create file '%s'", fpath)
|
beeLogger.Log.Warnf("Skipped create file '%s'", fpath)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
f, err = os.OpenFile(fpath, os.O_CREATE|os.O_RDWR, 0666)
|
f, err = os.OpenFile(fpath, os.O_CREATE|os.O_RDWR, 0666)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Warnf("%s", err)
|
beeLogger.Log.Warnf("%s", err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fileStr := strings.Replace(CtrlTPL, "{{ctrlName}}", camelCase(tb.Name), -1)
|
fileStr := strings.Replace(CtrlTPL, "{{ctrlName}}", utils.CamelCase(tb.Name), -1)
|
||||||
fileStr = strings.Replace(fileStr, "{{pkgPath}}", pkgPath, -1)
|
fileStr = strings.Replace(fileStr, "{{pkgPath}}", pkgPath, -1)
|
||||||
if _, err := f.WriteString(fileStr); err != nil {
|
if _, err := f.WriteString(fileStr); err != nil {
|
||||||
logger.Fatalf("Could not write controller file to '%s': %s", fpath, err)
|
beeLogger.Log.Fatalf("Could not write controller file to '%s': %s", fpath, err)
|
||||||
}
|
}
|
||||||
CloseFile(f)
|
utils.CloseFile(f)
|
||||||
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", fpath, "\x1b[0m")
|
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", fpath, "\x1b[0m")
|
||||||
formatSourceCode(fpath)
|
utils.FormatSourceCode(fpath)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// writeRouterFile generates router file
|
// writeRouterFile generates router file
|
||||||
func writeRouterFile(tables []*Table, rPath string, selectedTables map[string]bool, pkgPath string) {
|
func writeRouterFile(tables []*Table, rPath string, selectedTables map[string]bool, pkgPath string) {
|
||||||
w := NewColorWriter(os.Stdout)
|
w := colors.NewColorWriter(os.Stdout)
|
||||||
|
|
||||||
var nameSpaces []string
|
var nameSpaces []string
|
||||||
for _, tb := range tables {
|
for _, tb := range tables {
|
||||||
@ -856,7 +859,7 @@ func writeRouterFile(tables []*Table, rPath string, selectedTables map[string]bo
|
|||||||
}
|
}
|
||||||
// Add namespaces
|
// Add namespaces
|
||||||
nameSpace := strings.Replace(NamespaceTPL, "{{nameSpace}}", tb.Name, -1)
|
nameSpace := strings.Replace(NamespaceTPL, "{{nameSpace}}", tb.Name, -1)
|
||||||
nameSpace = strings.Replace(nameSpace, "{{ctrlName}}", camelCase(tb.Name), -1)
|
nameSpace = strings.Replace(nameSpace, "{{ctrlName}}", utils.CamelCase(tb.Name), -1)
|
||||||
nameSpaces = append(nameSpaces, nameSpace)
|
nameSpaces = append(nameSpaces, nameSpace)
|
||||||
}
|
}
|
||||||
// Add export controller
|
// Add export controller
|
||||||
@ -865,31 +868,31 @@ func writeRouterFile(tables []*Table, rPath string, selectedTables map[string]bo
|
|||||||
routerStr = strings.Replace(routerStr, "{{pkgPath}}", pkgPath, 1)
|
routerStr = strings.Replace(routerStr, "{{pkgPath}}", pkgPath, 1)
|
||||||
var f *os.File
|
var f *os.File
|
||||||
var err error
|
var err error
|
||||||
if isExist(fpath) {
|
if utils.IsExist(fpath) {
|
||||||
logger.Warnf("'%s' already exists. Do you want to overwrite it? [Yes|No] ", fpath)
|
beeLogger.Log.Warnf("'%s' already exists. Do you want to overwrite it? [Yes|No] ", fpath)
|
||||||
if askForConfirmation() {
|
if utils.AskForConfirmation() {
|
||||||
f, err = os.OpenFile(fpath, os.O_RDWR|os.O_TRUNC, 0666)
|
f, err = os.OpenFile(fpath, os.O_RDWR|os.O_TRUNC, 0666)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Warnf("%s", err)
|
beeLogger.Log.Warnf("%s", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
logger.Warnf("Skipped create file '%s'", fpath)
|
beeLogger.Log.Warnf("Skipped create file '%s'", fpath)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
f, err = os.OpenFile(fpath, os.O_CREATE|os.O_RDWR, 0666)
|
f, err = os.OpenFile(fpath, os.O_CREATE|os.O_RDWR, 0666)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Warnf("%s", err)
|
beeLogger.Log.Warnf("%s", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if _, err := f.WriteString(routerStr); err != nil {
|
if _, err := f.WriteString(routerStr); err != nil {
|
||||||
logger.Fatalf("Could not write router file to '%s': %s", fpath, err)
|
beeLogger.Log.Fatalf("Could not write router file to '%s': %s", fpath, err)
|
||||||
}
|
}
|
||||||
CloseFile(f)
|
utils.CloseFile(f)
|
||||||
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", fpath, "\x1b[0m")
|
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", fpath, "\x1b[0m")
|
||||||
formatSourceCode(fpath)
|
utils.FormatSourceCode(fpath)
|
||||||
}
|
}
|
||||||
|
|
||||||
func isSQLTemporalType(t string) bool {
|
func isSQLTemporalType(t string) bool {
|
||||||
@ -952,10 +955,10 @@ func getFileName(tbName string) (filename string) {
|
|||||||
func getPackagePath(curpath string) (packpath string) {
|
func getPackagePath(curpath string) (packpath string) {
|
||||||
gopath := os.Getenv("GOPATH")
|
gopath := os.Getenv("GOPATH")
|
||||||
if gopath == "" {
|
if gopath == "" {
|
||||||
logger.Fatal("GOPATH environment variable is not set or empty")
|
beeLogger.Log.Fatal("GOPATH environment variable is not set or empty")
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.Debugf("GOPATH: %s", __FILE__(), __LINE__(), gopath)
|
beeLogger.Log.Debugf("GOPATH: %s", utils.FILE(), utils.LINE(), gopath)
|
||||||
|
|
||||||
appsrcpath := ""
|
appsrcpath := ""
|
||||||
haspath := false
|
haspath := false
|
||||||
@ -964,7 +967,7 @@ func getPackagePath(curpath string) (packpath string) {
|
|||||||
for _, wg := range wgopath {
|
for _, wg := range wgopath {
|
||||||
wg, _ = filepath.EvalSymlinks(path.Join(wg, "src"))
|
wg, _ = filepath.EvalSymlinks(path.Join(wg, "src"))
|
||||||
|
|
||||||
if filepath.HasPrefix(strings.ToLower(curpath), strings.ToLower(wg)) {
|
if strings.HasPrefix(strings.ToLower(curpath), strings.ToLower(wg)) {
|
||||||
haspath = true
|
haspath = true
|
||||||
appsrcpath = wg
|
appsrcpath = wg
|
||||||
break
|
break
|
||||||
@ -972,11 +975,11 @@ func getPackagePath(curpath string) (packpath string) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if !haspath {
|
if !haspath {
|
||||||
logger.Fatalf("Cannot generate application code outside of GOPATH '%s'", gopath)
|
beeLogger.Log.Fatalf("Cannot generate application code outside of GOPATH '%s'", gopath)
|
||||||
}
|
}
|
||||||
|
|
||||||
if curpath == appsrcpath {
|
if curpath == appsrcpath {
|
||||||
logger.Fatal("Cannot generate application code outside of application path")
|
beeLogger.Log.Fatal("Cannot generate application code outside of application path")
|
||||||
}
|
}
|
||||||
|
|
||||||
packpath = strings.Join(strings.Split(curpath[len(appsrcpath)+1:], string(filepath.Separator)), "/")
|
packpath = strings.Join(strings.Split(curpath[len(appsrcpath)+1:], string(filepath.Separator)), "/")
|
@ -12,17 +12,21 @@
|
|||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
|
||||||
package main
|
package generate
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/logger/colors"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
func generateController(cname, currpath string) {
|
func GenerateController(cname, currpath string) {
|
||||||
w := NewColorWriter(os.Stdout)
|
w := colors.NewColorWriter(os.Stdout)
|
||||||
|
|
||||||
p, f := path.Split(cname)
|
p, f := path.Split(cname)
|
||||||
controllerName := strings.Title(f)
|
controllerName := strings.Title(f)
|
||||||
@ -33,26 +37,26 @@ func generateController(cname, currpath string) {
|
|||||||
packageName = p[i+1 : len(p)-1]
|
packageName = p[i+1 : len(p)-1]
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.Infof("Using '%s' as controller name", controllerName)
|
beeLogger.Log.Infof("Using '%s' as controller name", controllerName)
|
||||||
logger.Infof("Using '%s' as package name", packageName)
|
beeLogger.Log.Infof("Using '%s' as package name", packageName)
|
||||||
|
|
||||||
fp := path.Join(currpath, "controllers", p)
|
fp := path.Join(currpath, "controllers", p)
|
||||||
if _, err := os.Stat(fp); os.IsNotExist(err) {
|
if _, err := os.Stat(fp); os.IsNotExist(err) {
|
||||||
// Create the controller's directory
|
// Create the controller's directory
|
||||||
if err := os.MkdirAll(fp, 0777); err != nil {
|
if err := os.MkdirAll(fp, 0777); err != nil {
|
||||||
logger.Fatalf("Could not create controllers directory: %s", err)
|
beeLogger.Log.Fatalf("Could not create controllers directory: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fpath := path.Join(fp, strings.ToLower(controllerName)+".go")
|
fpath := path.Join(fp, strings.ToLower(controllerName)+".go")
|
||||||
if f, err := os.OpenFile(fpath, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err == nil {
|
if f, err := os.OpenFile(fpath, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err == nil {
|
||||||
defer CloseFile(f)
|
defer utils.CloseFile(f)
|
||||||
|
|
||||||
modelPath := path.Join(currpath, "models", strings.ToLower(controllerName)+".go")
|
modelPath := path.Join(currpath, "models", strings.ToLower(controllerName)+".go")
|
||||||
|
|
||||||
var content string
|
var content string
|
||||||
if _, err := os.Stat(modelPath); err == nil {
|
if _, err := os.Stat(modelPath); err == nil {
|
||||||
logger.Infof("Using matching model '%s'", controllerName)
|
beeLogger.Log.Infof("Using matching model '%s'", controllerName)
|
||||||
content = strings.Replace(controllerModelTpl, "{{packageName}}", packageName, -1)
|
content = strings.Replace(controllerModelTpl, "{{packageName}}", packageName, -1)
|
||||||
pkgPath := getPackagePath(currpath)
|
pkgPath := getPackagePath(currpath)
|
||||||
content = strings.Replace(content, "{{pkgPath}}", pkgPath, -1)
|
content = strings.Replace(content, "{{pkgPath}}", pkgPath, -1)
|
||||||
@ -64,10 +68,10 @@ func generateController(cname, currpath string) {
|
|||||||
f.WriteString(content)
|
f.WriteString(content)
|
||||||
|
|
||||||
// Run 'gofmt' on the generated source code
|
// Run 'gofmt' on the generated source code
|
||||||
formatSourceCode(fpath)
|
utils.FormatSourceCode(fpath)
|
||||||
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", fpath, "\x1b[0m")
|
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", fpath, "\x1b[0m")
|
||||||
} else {
|
} else {
|
||||||
logger.Fatalf("Could not create controller file: %s", err)
|
beeLogger.Log.Fatalf("Could not create controller file: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -15,7 +15,7 @@
|
|||||||
* *
|
* *
|
||||||
\**********************************************************/
|
\**********************************************************/
|
||||||
|
|
||||||
package main
|
package generate
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"database/sql"
|
"database/sql"
|
||||||
@ -24,11 +24,256 @@ import (
|
|||||||
"path"
|
"path"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/logger/colors"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
_ "github.com/go-sql-driver/mysql"
|
_ "github.com/go-sql-driver/mysql"
|
||||||
_ "github.com/lib/pq"
|
_ "github.com/lib/pq"
|
||||||
)
|
)
|
||||||
|
|
||||||
func generateHproseAppcode(driver, connStr, level, tables, currpath string) {
|
var Hproseconf = `appname = {{.Appname}}
|
||||||
|
httpport = 8080
|
||||||
|
runmode = dev
|
||||||
|
autorender = false
|
||||||
|
copyrequestbody = true
|
||||||
|
EnableDocs = true
|
||||||
|
`
|
||||||
|
var HproseMaingo = `package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"{{.Appname}}/models"
|
||||||
|
"github.com/hprose/hprose-golang/rpc"
|
||||||
|
|
||||||
|
"github.com/astaxie/beego"
|
||||||
|
)
|
||||||
|
|
||||||
|
func logInvokeHandler(
|
||||||
|
name string,
|
||||||
|
args []reflect.Value,
|
||||||
|
context rpc.Context,
|
||||||
|
next rpc.NextInvokeHandler) (results []reflect.Value, err error) {
|
||||||
|
fmt.Printf("%s(%v) = ", name, args)
|
||||||
|
results, err = next(name, args, context)
|
||||||
|
fmt.Printf("%v %v\r\n", results, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
// Create WebSocketServer
|
||||||
|
// service := rpc.NewWebSocketService()
|
||||||
|
|
||||||
|
// Create Http Server
|
||||||
|
service := rpc.NewHTTPService()
|
||||||
|
|
||||||
|
// Use Logger Middleware
|
||||||
|
service.AddInvokeHandler(logInvokeHandler)
|
||||||
|
|
||||||
|
// Publish Functions
|
||||||
|
service.AddFunction("AddOne", models.AddOne)
|
||||||
|
service.AddFunction("GetOne", models.GetOne)
|
||||||
|
|
||||||
|
// Start Service
|
||||||
|
beego.Handler("/", service)
|
||||||
|
beego.Run()
|
||||||
|
}
|
||||||
|
`
|
||||||
|
|
||||||
|
var HproseMainconngo = `package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"{{.Appname}}/models"
|
||||||
|
"github.com/hprose/hprose-golang/rpc"
|
||||||
|
|
||||||
|
"github.com/astaxie/beego"
|
||||||
|
"github.com/astaxie/beego/orm"
|
||||||
|
{{.DriverPkg}}
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
orm.RegisterDataBase("default", "{{.DriverName}}", "{{.conn}}")
|
||||||
|
}
|
||||||
|
|
||||||
|
func logInvokeHandler(
|
||||||
|
name string,
|
||||||
|
args []reflect.Value,
|
||||||
|
context rpc.Context,
|
||||||
|
next rpc.NextInvokeHandler) (results []reflect.Value, err error) {
|
||||||
|
fmt.Printf("%s(%v) = ", name, args)
|
||||||
|
results, err = next(name, args, context)
|
||||||
|
fmt.Printf("%v %v\r\n", results, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
// Create WebSocketServer
|
||||||
|
// service := rpc.NewWebSocketService()
|
||||||
|
|
||||||
|
// Create Http Server
|
||||||
|
service := rpc.NewHTTPService()
|
||||||
|
|
||||||
|
// Use Logger Middleware
|
||||||
|
service.AddInvokeHandler(logInvokeHandler)
|
||||||
|
|
||||||
|
{{HproseFunctionList}}
|
||||||
|
|
||||||
|
// Start Service
|
||||||
|
beego.Handler("/", service)
|
||||||
|
beego.Run()
|
||||||
|
}
|
||||||
|
|
||||||
|
`
|
||||||
|
|
||||||
|
var HproseModels = `package models
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"strconv"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
Objects map[string]*Object
|
||||||
|
)
|
||||||
|
|
||||||
|
type Object struct {
|
||||||
|
ObjectId string
|
||||||
|
Score int64
|
||||||
|
PlayerName string
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
Objects = make(map[string]*Object)
|
||||||
|
Objects["hjkhsbnmn123"] = &Object{"hjkhsbnmn123", 100, "astaxie"}
|
||||||
|
Objects["mjjkxsxsaa23"] = &Object{"mjjkxsxsaa23", 101, "someone"}
|
||||||
|
}
|
||||||
|
|
||||||
|
func AddOne(object Object) (ObjectId string) {
|
||||||
|
object.ObjectId = "astaxie" + strconv.FormatInt(time.Now().UnixNano(), 10)
|
||||||
|
Objects[object.ObjectId] = &object
|
||||||
|
return object.ObjectId
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetOne(ObjectId string) (object *Object, err error) {
|
||||||
|
if v, ok := Objects[ObjectId]; ok {
|
||||||
|
return v, nil
|
||||||
|
}
|
||||||
|
return nil, errors.New("ObjectId Not Exist")
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetAll() map[string]*Object {
|
||||||
|
return Objects
|
||||||
|
}
|
||||||
|
|
||||||
|
func Update(ObjectId string, Score int64) (err error) {
|
||||||
|
if v, ok := Objects[ObjectId]; ok {
|
||||||
|
v.Score = Score
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return errors.New("ObjectId Not Exist")
|
||||||
|
}
|
||||||
|
|
||||||
|
func Delete(ObjectId string) {
|
||||||
|
delete(Objects, ObjectId)
|
||||||
|
}
|
||||||
|
|
||||||
|
`
|
||||||
|
|
||||||
|
var HproseModels2 = `package models
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"strconv"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
UserList map[string]*User
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
UserList = make(map[string]*User)
|
||||||
|
u := User{"user_11111", "astaxie", "11111", Profile{"male", 20, "Singapore", "astaxie@gmail.com"}}
|
||||||
|
UserList["user_11111"] = &u
|
||||||
|
}
|
||||||
|
|
||||||
|
type User struct {
|
||||||
|
Id string
|
||||||
|
Username string
|
||||||
|
Password string
|
||||||
|
Profile Profile
|
||||||
|
}
|
||||||
|
|
||||||
|
type Profile struct {
|
||||||
|
Gender string
|
||||||
|
Age int
|
||||||
|
Address string
|
||||||
|
Email string
|
||||||
|
}
|
||||||
|
|
||||||
|
func AddUser(u User) string {
|
||||||
|
u.Id = "user_" + strconv.FormatInt(time.Now().UnixNano(), 10)
|
||||||
|
UserList[u.Id] = &u
|
||||||
|
return u.Id
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetUser(uid string) (u *User, err error) {
|
||||||
|
if u, ok := UserList[uid]; ok {
|
||||||
|
return u, nil
|
||||||
|
}
|
||||||
|
return nil, errors.New("User not exists")
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetAllUsers() map[string]*User {
|
||||||
|
return UserList
|
||||||
|
}
|
||||||
|
|
||||||
|
func UpdateUser(uid string, uu *User) (a *User, err error) {
|
||||||
|
if u, ok := UserList[uid]; ok {
|
||||||
|
if uu.Username != "" {
|
||||||
|
u.Username = uu.Username
|
||||||
|
}
|
||||||
|
if uu.Password != "" {
|
||||||
|
u.Password = uu.Password
|
||||||
|
}
|
||||||
|
if uu.Profile.Age != 0 {
|
||||||
|
u.Profile.Age = uu.Profile.Age
|
||||||
|
}
|
||||||
|
if uu.Profile.Address != "" {
|
||||||
|
u.Profile.Address = uu.Profile.Address
|
||||||
|
}
|
||||||
|
if uu.Profile.Gender != "" {
|
||||||
|
u.Profile.Gender = uu.Profile.Gender
|
||||||
|
}
|
||||||
|
if uu.Profile.Email != "" {
|
||||||
|
u.Profile.Email = uu.Profile.Email
|
||||||
|
}
|
||||||
|
return u, nil
|
||||||
|
}
|
||||||
|
return nil, errors.New("User Not Exist")
|
||||||
|
}
|
||||||
|
|
||||||
|
func Login(username, password string) bool {
|
||||||
|
for _, u := range UserList {
|
||||||
|
if u.Username == username && u.Password == password {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func DeleteUser(uid string) {
|
||||||
|
delete(UserList, uid)
|
||||||
|
}
|
||||||
|
`
|
||||||
|
var HproseAddFunctions = []string{}
|
||||||
|
|
||||||
|
func GenerateHproseAppcode(driver, connStr, level, tables, currpath string) {
|
||||||
var mode byte
|
var mode byte
|
||||||
switch level {
|
switch level {
|
||||||
case "1":
|
case "1":
|
||||||
@ -38,7 +283,7 @@ func generateHproseAppcode(driver, connStr, level, tables, currpath string) {
|
|||||||
case "3":
|
case "3":
|
||||||
mode = OModel | OController | ORouter
|
mode = OModel | OController | ORouter
|
||||||
default:
|
default:
|
||||||
logger.Fatal("Invalid 'level' option. Level must be either \"1\", \"2\" or \"3\"")
|
beeLogger.Log.Fatal("Invalid 'level' option. Level must be either \"1\", \"2\" or \"3\"")
|
||||||
}
|
}
|
||||||
var selectedTables map[string]bool
|
var selectedTables map[string]bool
|
||||||
if tables != "" {
|
if tables != "" {
|
||||||
@ -51,9 +296,9 @@ func generateHproseAppcode(driver, connStr, level, tables, currpath string) {
|
|||||||
case "mysql":
|
case "mysql":
|
||||||
case "postgres":
|
case "postgres":
|
||||||
case "sqlite":
|
case "sqlite":
|
||||||
logger.Fatal("Generating app code from SQLite database is not supported yet")
|
beeLogger.Log.Fatal("Generating app code from SQLite database is not supported yet")
|
||||||
default:
|
default:
|
||||||
logger.Fatalf("Unknown database driver '%s'. Driver must be one of mysql, postgres or sqlite", driver)
|
beeLogger.Log.Fatalf("Unknown database driver '%s'. Driver must be one of mysql, postgres or sqlite", driver)
|
||||||
}
|
}
|
||||||
genHprose(driver, connStr, mode, selectedTables, currpath)
|
genHprose(driver, connStr, mode, selectedTables, currpath)
|
||||||
}
|
}
|
||||||
@ -63,11 +308,11 @@ func generateHproseAppcode(driver, connStr, level, tables, currpath string) {
|
|||||||
func genHprose(dbms, connStr string, mode byte, selectedTableNames map[string]bool, currpath string) {
|
func genHprose(dbms, connStr string, mode byte, selectedTableNames map[string]bool, currpath string) {
|
||||||
db, err := sql.Open(dbms, connStr)
|
db, err := sql.Open(dbms, connStr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Could not connect to '%s' database using '%s': %s", dbms, connStr, err)
|
beeLogger.Log.Fatalf("Could not connect to '%s' database using '%s': %s", dbms, connStr, err)
|
||||||
}
|
}
|
||||||
defer db.Close()
|
defer db.Close()
|
||||||
if trans, ok := dbDriver[dbms]; ok {
|
if trans, ok := dbDriver[dbms]; ok {
|
||||||
logger.Info("Analyzing database tables...")
|
beeLogger.Log.Info("Analyzing database tables...")
|
||||||
tableNames := trans.GetTableNames(db)
|
tableNames := trans.GetTableNames(db)
|
||||||
tables := getTableObjects(tableNames, db, trans)
|
tables := getTableObjects(tableNames, db, trans)
|
||||||
mvcPath := new(MvcPath)
|
mvcPath := new(MvcPath)
|
||||||
@ -76,7 +321,7 @@ func genHprose(dbms, connStr string, mode byte, selectedTableNames map[string]bo
|
|||||||
pkgPath := getPackagePath(currpath)
|
pkgPath := getPackagePath(currpath)
|
||||||
writeHproseSourceFiles(pkgPath, tables, mode, mvcPath, selectedTableNames)
|
writeHproseSourceFiles(pkgPath, tables, mode, mvcPath, selectedTableNames)
|
||||||
} else {
|
} else {
|
||||||
logger.Fatalf("Generating app code from '%s' database is not supported yet", dbms)
|
beeLogger.Log.Fatalf("Generating app code from '%s' database is not supported yet", dbms)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -85,14 +330,14 @@ func genHprose(dbms, connStr string, mode byte, selectedTableNames map[string]bo
|
|||||||
// Newly geneated files will be inside these folders.
|
// Newly geneated files will be inside these folders.
|
||||||
func writeHproseSourceFiles(pkgPath string, tables []*Table, mode byte, paths *MvcPath, selectedTables map[string]bool) {
|
func writeHproseSourceFiles(pkgPath string, tables []*Table, mode byte, paths *MvcPath, selectedTables map[string]bool) {
|
||||||
if (OModel & mode) == OModel {
|
if (OModel & mode) == OModel {
|
||||||
logger.Info("Creating model files...")
|
beeLogger.Log.Info("Creating model files...")
|
||||||
writeHproseModelFiles(tables, paths.ModelPath, selectedTables)
|
writeHproseModelFiles(tables, paths.ModelPath, selectedTables)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// writeHproseModelFiles generates model files
|
// writeHproseModelFiles generates model files
|
||||||
func writeHproseModelFiles(tables []*Table, mPath string, selectedTables map[string]bool) {
|
func writeHproseModelFiles(tables []*Table, mPath string, selectedTables map[string]bool) {
|
||||||
w := NewColorWriter(os.Stdout)
|
w := colors.NewColorWriter(os.Stdout)
|
||||||
|
|
||||||
for _, tb := range tables {
|
for _, tb := range tables {
|
||||||
// if selectedTables map is not nil and this table is not selected, ignore it
|
// if selectedTables map is not nil and this table is not selected, ignore it
|
||||||
@ -105,22 +350,22 @@ func writeHproseModelFiles(tables []*Table, mPath string, selectedTables map[str
|
|||||||
fpath := path.Join(mPath, filename+".go")
|
fpath := path.Join(mPath, filename+".go")
|
||||||
var f *os.File
|
var f *os.File
|
||||||
var err error
|
var err error
|
||||||
if isExist(fpath) {
|
if utils.IsExist(fpath) {
|
||||||
logger.Warnf("'%s' already exists. Do you want to overwrite it? [Yes|No] ", fpath)
|
beeLogger.Log.Warnf("'%s' already exists. Do you want to overwrite it? [Yes|No] ", fpath)
|
||||||
if askForConfirmation() {
|
if utils.AskForConfirmation() {
|
||||||
f, err = os.OpenFile(fpath, os.O_RDWR|os.O_TRUNC, 0666)
|
f, err = os.OpenFile(fpath, os.O_RDWR|os.O_TRUNC, 0666)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Warnf("%s", err)
|
beeLogger.Log.Warnf("%s", err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
logger.Warnf("Skipped create file '%s'", fpath)
|
beeLogger.Log.Warnf("Skipped create file '%s'", fpath)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
f, err = os.OpenFile(fpath, os.O_CREATE|os.O_RDWR, 0666)
|
f, err = os.OpenFile(fpath, os.O_CREATE|os.O_RDWR, 0666)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Warnf("%s", err)
|
beeLogger.Log.Warnf("%s", err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -129,10 +374,10 @@ func writeHproseModelFiles(tables []*Table, mPath string, selectedTables map[str
|
|||||||
template = HproseStructModelTPL
|
template = HproseStructModelTPL
|
||||||
} else {
|
} else {
|
||||||
template = HproseModelTPL
|
template = HproseModelTPL
|
||||||
hproseAddFunctions = append(hproseAddFunctions, strings.Replace(HproseAddFunction, "{{modelName}}", camelCase(tb.Name), -1))
|
HproseAddFunctions = append(HproseAddFunctions, strings.Replace(HproseAddFunction, "{{modelName}}", utils.CamelCase(tb.Name), -1))
|
||||||
}
|
}
|
||||||
fileStr := strings.Replace(template, "{{modelStruct}}", tb.String(), 1)
|
fileStr := strings.Replace(template, "{{modelStruct}}", tb.String(), 1)
|
||||||
fileStr = strings.Replace(fileStr, "{{modelName}}", camelCase(tb.Name), -1)
|
fileStr = strings.Replace(fileStr, "{{modelName}}", utils.CamelCase(tb.Name), -1)
|
||||||
// if table contains time field, import time.Time package
|
// if table contains time field, import time.Time package
|
||||||
timePkg := ""
|
timePkg := ""
|
||||||
importTimePkg := ""
|
importTimePkg := ""
|
||||||
@ -143,11 +388,11 @@ func writeHproseModelFiles(tables []*Table, mPath string, selectedTables map[str
|
|||||||
fileStr = strings.Replace(fileStr, "{{timePkg}}", timePkg, -1)
|
fileStr = strings.Replace(fileStr, "{{timePkg}}", timePkg, -1)
|
||||||
fileStr = strings.Replace(fileStr, "{{importTimePkg}}", importTimePkg, -1)
|
fileStr = strings.Replace(fileStr, "{{importTimePkg}}", importTimePkg, -1)
|
||||||
if _, err := f.WriteString(fileStr); err != nil {
|
if _, err := f.WriteString(fileStr); err != nil {
|
||||||
logger.Fatalf("Could not write model file to '%s'", fpath)
|
beeLogger.Log.Fatalf("Could not write model file to '%s'", fpath)
|
||||||
}
|
}
|
||||||
CloseFile(f)
|
utils.CloseFile(f)
|
||||||
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", fpath, "\x1b[0m")
|
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", fpath, "\x1b[0m")
|
||||||
formatSourceCode(fpath)
|
utils.FormatSourceCode(fpath)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -12,7 +12,7 @@
|
|||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
|
||||||
package main
|
package generate
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
@ -20,6 +20,10 @@ import (
|
|||||||
"path"
|
"path"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/logger/colors"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
@ -29,18 +33,18 @@ const (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type DBDriver interface {
|
type DBDriver interface {
|
||||||
generateCreateUp(tableName string) string
|
GenerateCreateUp(tableName string) string
|
||||||
generateCreateDown(tableName string) string
|
GenerateCreateDown(tableName string) string
|
||||||
}
|
}
|
||||||
|
|
||||||
type mysqlDriver struct{}
|
type mysqlDriver struct{}
|
||||||
|
|
||||||
func (m mysqlDriver) generateCreateUp(tableName string) string {
|
func (m mysqlDriver) GenerateCreateUp(tableName string) string {
|
||||||
upsql := `m.SQL("CREATE TABLE ` + tableName + "(" + m.generateSQLFromFields(fields.String()) + `)");`
|
upsql := `m.SQL("CREATE TABLE ` + tableName + "(" + m.generateSQLFromFields(Fields.String()) + `)");`
|
||||||
return upsql
|
return upsql
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m mysqlDriver) generateCreateDown(tableName string) string {
|
func (m mysqlDriver) GenerateCreateDown(tableName string) string {
|
||||||
downsql := `m.SQL("DROP TABLE ` + "`" + tableName + "`" + `")`
|
downsql := `m.SQL("DROP TABLE ` + "`" + tableName + "`" + `")`
|
||||||
return downsql
|
return downsql
|
||||||
}
|
}
|
||||||
@ -51,21 +55,21 @@ func (m mysqlDriver) generateSQLFromFields(fields string) string {
|
|||||||
for i, v := range fds {
|
for i, v := range fds {
|
||||||
kv := strings.SplitN(v, ":", 2)
|
kv := strings.SplitN(v, ":", 2)
|
||||||
if len(kv) != 2 {
|
if len(kv) != 2 {
|
||||||
logger.Error("Fields format is wrong. Should be: key:type,key:type " + v)
|
beeLogger.Log.Error("Fields format is wrong. Should be: key:type,key:type " + v)
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
typ, tag := m.getSQLType(kv[1])
|
typ, tag := m.getSQLType(kv[1])
|
||||||
if typ == "" {
|
if typ == "" {
|
||||||
logger.Error("Fields format is wrong. Should be: key:type,key:type " + v)
|
beeLogger.Log.Error("Fields format is wrong. Should be: key:type,key:type " + v)
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
if i == 0 && strings.ToLower(kv[0]) != "id" {
|
if i == 0 && strings.ToLower(kv[0]) != "id" {
|
||||||
sql += "`id` int(11) NOT NULL AUTO_INCREMENT,"
|
sql += "`id` int(11) NOT NULL AUTO_INCREMENT,"
|
||||||
tags = tags + "PRIMARY KEY (`id`),"
|
tags = tags + "PRIMARY KEY (`id`),"
|
||||||
}
|
}
|
||||||
sql += "`" + snakeString(kv[0]) + "` " + typ + ","
|
sql += "`" + utils.SnakeString(kv[0]) + "` " + typ + ","
|
||||||
if tag != "" {
|
if tag != "" {
|
||||||
tags = tags + fmt.Sprintf(tag, "`"+snakeString(kv[0])+"`") + ","
|
tags = tags + fmt.Sprintf(tag, "`"+utils.SnakeString(kv[0])+"`") + ","
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
sql = strings.TrimRight(sql+tags, ",")
|
sql = strings.TrimRight(sql+tags, ",")
|
||||||
@ -104,12 +108,12 @@ func (m mysqlDriver) getSQLType(ktype string) (tp, tag string) {
|
|||||||
|
|
||||||
type postgresqlDriver struct{}
|
type postgresqlDriver struct{}
|
||||||
|
|
||||||
func (m postgresqlDriver) generateCreateUp(tableName string) string {
|
func (m postgresqlDriver) GenerateCreateUp(tableName string) string {
|
||||||
upsql := `m.SQL("CREATE TABLE ` + tableName + "(" + m.generateSQLFromFields(fields.String()) + `)");`
|
upsql := `m.SQL("CREATE TABLE ` + tableName + "(" + m.generateSQLFromFields(Fields.String()) + `)");`
|
||||||
return upsql
|
return upsql
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m postgresqlDriver) generateCreateDown(tableName string) string {
|
func (m postgresqlDriver) GenerateCreateDown(tableName string) string {
|
||||||
downsql := `m.SQL("DROP TABLE ` + tableName + `")`
|
downsql := `m.SQL("DROP TABLE ` + tableName + `")`
|
||||||
return downsql
|
return downsql
|
||||||
}
|
}
|
||||||
@ -120,20 +124,20 @@ func (m postgresqlDriver) generateSQLFromFields(fields string) string {
|
|||||||
for i, v := range fds {
|
for i, v := range fds {
|
||||||
kv := strings.SplitN(v, ":", 2)
|
kv := strings.SplitN(v, ":", 2)
|
||||||
if len(kv) != 2 {
|
if len(kv) != 2 {
|
||||||
logger.Error("Fields format is wrong. Should be: key:type,key:type " + v)
|
beeLogger.Log.Error("Fields format is wrong. Should be: key:type,key:type " + v)
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
typ, tag := m.getSQLType(kv[1])
|
typ, tag := m.getSQLType(kv[1])
|
||||||
if typ == "" {
|
if typ == "" {
|
||||||
logger.Error("Fields format is wrong. Should be: key:type,key:type " + v)
|
beeLogger.Log.Error("Fields format is wrong. Should be: key:type,key:type " + v)
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
if i == 0 && strings.ToLower(kv[0]) != "id" {
|
if i == 0 && strings.ToLower(kv[0]) != "id" {
|
||||||
sql += "id serial primary key,"
|
sql += "id serial primary key,"
|
||||||
}
|
}
|
||||||
sql += snakeString(kv[0]) + " " + typ + ","
|
sql += utils.SnakeString(kv[0]) + " " + typ + ","
|
||||||
if tag != "" {
|
if tag != "" {
|
||||||
tags = tags + fmt.Sprintf(tag, snakeString(kv[0])) + ","
|
tags = tags + fmt.Sprintf(tag, utils.SnakeString(kv[0])) + ","
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if tags != "" {
|
if tags != "" {
|
||||||
@ -170,14 +174,14 @@ func (m postgresqlDriver) getSQLType(ktype string) (tp, tag string) {
|
|||||||
return "", ""
|
return "", ""
|
||||||
}
|
}
|
||||||
|
|
||||||
func newDBDriver() DBDriver {
|
func NewDBDriver() DBDriver {
|
||||||
switch driver {
|
switch SQLDriver {
|
||||||
case "mysql":
|
case "mysql":
|
||||||
return mysqlDriver{}
|
return mysqlDriver{}
|
||||||
case "postgres":
|
case "postgres":
|
||||||
return postgresqlDriver{}
|
return postgresqlDriver{}
|
||||||
default:
|
default:
|
||||||
logger.Fatal("Driver not supported")
|
beeLogger.Log.Fatal("Driver not supported")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -185,30 +189,30 @@ func newDBDriver() DBDriver {
|
|||||||
// generateMigration generates migration file template for database schema update.
|
// generateMigration generates migration file template for database schema update.
|
||||||
// The generated file template consists of an up() method for updating schema and
|
// The generated file template consists of an up() method for updating schema and
|
||||||
// a down() method for reverting the update.
|
// a down() method for reverting the update.
|
||||||
func generateMigration(mname, upsql, downsql, curpath string) {
|
func GenerateMigration(mname, upsql, downsql, curpath string) {
|
||||||
w := NewColorWriter(os.Stdout)
|
w := colors.NewColorWriter(os.Stdout)
|
||||||
migrationFilePath := path.Join(curpath, DBPath, MPath)
|
migrationFilePath := path.Join(curpath, DBPath, MPath)
|
||||||
if _, err := os.Stat(migrationFilePath); os.IsNotExist(err) {
|
if _, err := os.Stat(migrationFilePath); os.IsNotExist(err) {
|
||||||
// create migrations directory
|
// create migrations directory
|
||||||
if err := os.MkdirAll(migrationFilePath, 0777); err != nil {
|
if err := os.MkdirAll(migrationFilePath, 0777); err != nil {
|
||||||
logger.Fatalf("Could not create migration directory: %s", err)
|
beeLogger.Log.Fatalf("Could not create migration directory: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// create file
|
// create file
|
||||||
today := time.Now().Format(MDateFormat)
|
today := time.Now().Format(MDateFormat)
|
||||||
fpath := path.Join(migrationFilePath, fmt.Sprintf("%s_%s.go", today, mname))
|
fpath := path.Join(migrationFilePath, fmt.Sprintf("%s_%s.go", today, mname))
|
||||||
if f, err := os.OpenFile(fpath, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err == nil {
|
if f, err := os.OpenFile(fpath, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err == nil {
|
||||||
defer CloseFile(f)
|
defer utils.CloseFile(f)
|
||||||
content := strings.Replace(MigrationTPL, "{{StructName}}", camelCase(mname)+"_"+today, -1)
|
content := strings.Replace(MigrationTPL, "{{StructName}}", utils.CamelCase(mname)+"_"+today, -1)
|
||||||
content = strings.Replace(content, "{{CurrTime}}", today, -1)
|
content = strings.Replace(content, "{{CurrTime}}", today, -1)
|
||||||
content = strings.Replace(content, "{{UpSQL}}", upsql, -1)
|
content = strings.Replace(content, "{{UpSQL}}", upsql, -1)
|
||||||
content = strings.Replace(content, "{{DownSQL}}", downsql, -1)
|
content = strings.Replace(content, "{{DownSQL}}", downsql, -1)
|
||||||
f.WriteString(content)
|
f.WriteString(content)
|
||||||
// Run 'gofmt' on the generated source code
|
// Run 'gofmt' on the generated source code
|
||||||
formatSourceCode(fpath)
|
utils.FormatSourceCode(fpath)
|
||||||
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", fpath, "\x1b[0m")
|
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", fpath, "\x1b[0m")
|
||||||
} else {
|
} else {
|
||||||
logger.Fatalf("Could not create migration file: %s", err)
|
beeLogger.Log.Fatalf("Could not create migration file: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -12,7 +12,7 @@
|
|||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
|
||||||
package main
|
package generate
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
@ -20,10 +20,14 @@ import (
|
|||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/logger/colors"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
func generateModel(mname, fields, currpath string) {
|
func GenerateModel(mname, fields, currpath string) {
|
||||||
w := NewColorWriter(os.Stdout)
|
w := colors.NewColorWriter(os.Stdout)
|
||||||
|
|
||||||
p, f := path.Split(mname)
|
p, f := path.Split(mname)
|
||||||
modelName := strings.Title(f)
|
modelName := strings.Title(f)
|
||||||
@ -35,23 +39,23 @@ func generateModel(mname, fields, currpath string) {
|
|||||||
|
|
||||||
modelStruct, hastime, err := getStruct(modelName, fields)
|
modelStruct, hastime, err := getStruct(modelName, fields)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Could not generate the model struct: %s", err)
|
beeLogger.Log.Fatalf("Could not generate the model struct: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.Infof("Using '%s' as model name", modelName)
|
beeLogger.Log.Infof("Using '%s' as model name", modelName)
|
||||||
logger.Infof("Using '%s' as package name", packageName)
|
beeLogger.Log.Infof("Using '%s' as package name", packageName)
|
||||||
|
|
||||||
fp := path.Join(currpath, "models", p)
|
fp := path.Join(currpath, "models", p)
|
||||||
if _, err := os.Stat(fp); os.IsNotExist(err) {
|
if _, err := os.Stat(fp); os.IsNotExist(err) {
|
||||||
// Create the model's directory
|
// Create the model's directory
|
||||||
if err := os.MkdirAll(fp, 0777); err != nil {
|
if err := os.MkdirAll(fp, 0777); err != nil {
|
||||||
logger.Fatalf("Could not create the model directory: %s", err)
|
beeLogger.Log.Fatalf("Could not create the model directory: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fpath := path.Join(fp, strings.ToLower(modelName)+".go")
|
fpath := path.Join(fp, strings.ToLower(modelName)+".go")
|
||||||
if f, err := os.OpenFile(fpath, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err == nil {
|
if f, err := os.OpenFile(fpath, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err == nil {
|
||||||
defer CloseFile(f)
|
defer utils.CloseFile(f)
|
||||||
content := strings.Replace(modelTpl, "{{packageName}}", packageName, -1)
|
content := strings.Replace(modelTpl, "{{packageName}}", packageName, -1)
|
||||||
content = strings.Replace(content, "{{modelName}}", modelName, -1)
|
content = strings.Replace(content, "{{modelName}}", modelName, -1)
|
||||||
content = strings.Replace(content, "{{modelStruct}}", modelStruct, -1)
|
content = strings.Replace(content, "{{modelStruct}}", modelStruct, -1)
|
||||||
@ -62,10 +66,10 @@ func generateModel(mname, fields, currpath string) {
|
|||||||
}
|
}
|
||||||
f.WriteString(content)
|
f.WriteString(content)
|
||||||
// Run 'gofmt' on the generated source code
|
// Run 'gofmt' on the generated source code
|
||||||
formatSourceCode(fpath)
|
utils.FormatSourceCode(fpath)
|
||||||
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", fpath, "\x1b[0m")
|
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", fpath, "\x1b[0m")
|
||||||
} else {
|
} else {
|
||||||
logger.Fatalf("Could not create model file: %s", err)
|
beeLogger.Log.Fatalf("Could not create model file: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -95,7 +99,7 @@ func getStruct(structname, fields string) (string, bool, error) {
|
|||||||
if hastimeinner {
|
if hastimeinner {
|
||||||
hastime = true
|
hastime = true
|
||||||
}
|
}
|
||||||
structStr = structStr + camelString(kv[0]) + " " + typ + " " + tag + "\n"
|
structStr = structStr + utils.CamelString(kv[0]) + " " + typ + " " + tag + "\n"
|
||||||
}
|
}
|
||||||
structStr += "}\n"
|
structStr += "}\n"
|
||||||
return structStr, hastime, nil
|
return structStr, hastime, nil
|
50
generate/g_scaffold.go
Normal file
50
generate/g_scaffold.go
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
package generate
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd/commands/migrate"
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
|
)
|
||||||
|
|
||||||
|
func GenerateScaffold(sname, fields, currpath, driver, conn string) {
|
||||||
|
beeLogger.Log.Infof("Do you want to create a '%s' model? [Yes|No] ", sname)
|
||||||
|
|
||||||
|
// Generate the model
|
||||||
|
if utils.AskForConfirmation() {
|
||||||
|
GenerateModel(sname, fields, currpath)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate the controller
|
||||||
|
beeLogger.Log.Infof("Do you want to create a '%s' controller? [Yes|No] ", sname)
|
||||||
|
if utils.AskForConfirmation() {
|
||||||
|
GenerateController(sname, currpath)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate the views
|
||||||
|
beeLogger.Log.Infof("Do you want to create views for this '%s' resource? [Yes|No] ", sname)
|
||||||
|
if utils.AskForConfirmation() {
|
||||||
|
GenerateView(sname, currpath)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generate a migration
|
||||||
|
beeLogger.Log.Infof("Do you want to create a '%s' migration and schema for this resource? [Yes|No] ", sname)
|
||||||
|
if utils.AskForConfirmation() {
|
||||||
|
upsql := ""
|
||||||
|
downsql := ""
|
||||||
|
if fields != "" {
|
||||||
|
dbMigrator := NewDBDriver()
|
||||||
|
upsql = dbMigrator.GenerateCreateUp(sname)
|
||||||
|
downsql = dbMigrator.GenerateCreateDown(sname)
|
||||||
|
}
|
||||||
|
GenerateMigration(sname, upsql, downsql, currpath)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Run the migration
|
||||||
|
beeLogger.Log.Infof("Do you want to migrate the database? [Yes|No] ")
|
||||||
|
if utils.AskForConfirmation() {
|
||||||
|
migrate.MigrateUpdate(currpath, driver, conn)
|
||||||
|
}
|
||||||
|
beeLogger.Log.Successf("All done! Don't forget to add beego.Router(\"/%s\" ,&controllers.%sController{}) to routers/route.go\n", sname, strings.Title(sname))
|
||||||
|
}
|
@ -12,60 +12,64 @@
|
|||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
|
||||||
package main
|
package generate
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
|
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/logger/colors"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
// recipe
|
// recipe
|
||||||
// admin/recipe
|
// admin/recipe
|
||||||
func generateView(viewpath, currpath string) {
|
func GenerateView(viewpath, currpath string) {
|
||||||
w := NewColorWriter(os.Stdout)
|
w := colors.NewColorWriter(os.Stdout)
|
||||||
|
|
||||||
logger.Info("Generating view...")
|
beeLogger.Log.Info("Generating view...")
|
||||||
|
|
||||||
absViewPath := path.Join(currpath, "views", viewpath)
|
absViewPath := path.Join(currpath, "views", viewpath)
|
||||||
err := os.MkdirAll(absViewPath, os.ModePerm)
|
err := os.MkdirAll(absViewPath, os.ModePerm)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Could not create '%s' view: %s", viewpath, err)
|
beeLogger.Log.Fatalf("Could not create '%s' view: %s", viewpath, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
cfile := path.Join(absViewPath, "index.tpl")
|
cfile := path.Join(absViewPath, "index.tpl")
|
||||||
if f, err := os.OpenFile(cfile, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err == nil {
|
if f, err := os.OpenFile(cfile, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err == nil {
|
||||||
defer CloseFile(f)
|
defer utils.CloseFile(f)
|
||||||
f.WriteString(cfile)
|
f.WriteString(cfile)
|
||||||
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", cfile, "\x1b[0m")
|
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", cfile, "\x1b[0m")
|
||||||
} else {
|
} else {
|
||||||
logger.Fatalf("Could not create view file: %s", err)
|
beeLogger.Log.Fatalf("Could not create view file: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
cfile = path.Join(absViewPath, "show.tpl")
|
cfile = path.Join(absViewPath, "show.tpl")
|
||||||
if f, err := os.OpenFile(cfile, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err == nil {
|
if f, err := os.OpenFile(cfile, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err == nil {
|
||||||
defer CloseFile(f)
|
defer utils.CloseFile(f)
|
||||||
f.WriteString(cfile)
|
f.WriteString(cfile)
|
||||||
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", cfile, "\x1b[0m")
|
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", cfile, "\x1b[0m")
|
||||||
} else {
|
} else {
|
||||||
logger.Fatalf("Could not create view file: %s", err)
|
beeLogger.Log.Fatalf("Could not create view file: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
cfile = path.Join(absViewPath, "create.tpl")
|
cfile = path.Join(absViewPath, "create.tpl")
|
||||||
if f, err := os.OpenFile(cfile, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err == nil {
|
if f, err := os.OpenFile(cfile, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err == nil {
|
||||||
defer CloseFile(f)
|
defer utils.CloseFile(f)
|
||||||
f.WriteString(cfile)
|
f.WriteString(cfile)
|
||||||
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", cfile, "\x1b[0m")
|
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", cfile, "\x1b[0m")
|
||||||
} else {
|
} else {
|
||||||
logger.Fatalf("Could not create view file: %s", err)
|
beeLogger.Log.Fatalf("Could not create view file: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
cfile = path.Join(absViewPath, "edit.tpl")
|
cfile = path.Join(absViewPath, "edit.tpl")
|
||||||
if f, err := os.OpenFile(cfile, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err == nil {
|
if f, err := os.OpenFile(cfile, os.O_CREATE|os.O_EXCL|os.O_RDWR, 0666); err == nil {
|
||||||
defer CloseFile(f)
|
defer utils.CloseFile(f)
|
||||||
f.WriteString(cfile)
|
f.WriteString(cfile)
|
||||||
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", cfile, "\x1b[0m")
|
fmt.Fprintf(w, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", cfile, "\x1b[0m")
|
||||||
} else {
|
} else {
|
||||||
logger.Fatalf("Could not create view file: %s", err)
|
beeLogger.Log.Fatalf("Could not create view file: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -12,7 +12,7 @@
|
|||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
|
||||||
package main
|
package swaggergen
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
@ -35,6 +35,7 @@ import (
|
|||||||
|
|
||||||
"github.com/astaxie/beego/swagger"
|
"github.com/astaxie/beego/swagger"
|
||||||
"github.com/astaxie/beego/utils"
|
"github.com/astaxie/beego/utils"
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
@ -74,7 +75,7 @@ var basicTypes = map[string]string{
|
|||||||
"byte": "string:byte",
|
"byte": "string:byte",
|
||||||
"rune": "string:byte",
|
"rune": "string:byte",
|
||||||
// builtin golang objects
|
// builtin golang objects
|
||||||
"time.Time": "string:string",
|
"time.Time": "string:string",
|
||||||
}
|
}
|
||||||
|
|
||||||
var stdlibObject = map[string]string{
|
var stdlibObject = map[string]string{
|
||||||
@ -90,7 +91,7 @@ func init() {
|
|||||||
astPkgs = map[string]*ast.Package{}
|
astPkgs = map[string]*ast.Package{}
|
||||||
}
|
}
|
||||||
|
|
||||||
func parsePackagesFromDir(dirpath string) {
|
func ParsePackagesFromDir(dirpath string) {
|
||||||
c := make(chan error)
|
c := make(chan error)
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
@ -116,7 +117,7 @@ func parsePackagesFromDir(dirpath string) {
|
|||||||
}()
|
}()
|
||||||
|
|
||||||
for err := range c {
|
for err := range c {
|
||||||
logger.Warnf("%s", err)
|
beeLogger.Log.Warnf("%s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -137,12 +138,12 @@ func parsePackageFromDir(path string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func generateDocs(curpath string) {
|
func GenerateDocs(curpath string) {
|
||||||
fset := token.NewFileSet()
|
fset := token.NewFileSet()
|
||||||
|
|
||||||
f, err := parser.ParseFile(fset, path.Join(curpath, "routers", "router.go"), nil, parser.ParseComments)
|
f, err := parser.ParseFile(fset, path.Join(curpath, "routers", "router.go"), nil, parser.ParseComments)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Error while parsing router.go: %s", err)
|
beeLogger.Log.Fatalf("Error while parsing router.go: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
rootapi.Infos = swagger.Information{}
|
rootapi.Infos = swagger.Information{}
|
||||||
@ -251,6 +252,9 @@ func generateDocs(curpath string) {
|
|||||||
}
|
}
|
||||||
os.Mkdir(path.Join(curpath, "swagger"), 0755)
|
os.Mkdir(path.Join(curpath, "swagger"), 0755)
|
||||||
fd, err := os.Create(path.Join(curpath, "swagger", "swagger.json"))
|
fd, err := os.Create(path.Join(curpath, "swagger", "swagger.json"))
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
fdyml, err := os.Create(path.Join(curpath, "swagger", "swagger.yml"))
|
fdyml, err := os.Create(path.Join(curpath, "swagger", "swagger.yml"))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
@ -348,7 +352,7 @@ func analyseControllerPkg(localName, pkgpath string) {
|
|||||||
}
|
}
|
||||||
gopath := os.Getenv("GOPATH")
|
gopath := os.Getenv("GOPATH")
|
||||||
if gopath == "" {
|
if gopath == "" {
|
||||||
logger.Fatal("GOPATH environment variable is not set or empty")
|
beeLogger.Log.Fatal("GOPATH environment variable is not set or empty")
|
||||||
}
|
}
|
||||||
pkgRealpath := ""
|
pkgRealpath := ""
|
||||||
|
|
||||||
@ -366,7 +370,7 @@ func analyseControllerPkg(localName, pkgpath string) {
|
|||||||
}
|
}
|
||||||
pkgCache[pkgpath] = struct{}{}
|
pkgCache[pkgpath] = struct{}{}
|
||||||
} else {
|
} else {
|
||||||
logger.Fatalf("Package '%s' does not exist in the GOPATH", pkgpath)
|
beeLogger.Log.Fatalf("Package '%s' does not exist in the GOPATH", pkgpath)
|
||||||
}
|
}
|
||||||
|
|
||||||
fileSet := token.NewFileSet()
|
fileSet := token.NewFileSet()
|
||||||
@ -375,7 +379,7 @@ func analyseControllerPkg(localName, pkgpath string) {
|
|||||||
return !info.IsDir() && !strings.HasPrefix(name, ".") && strings.HasSuffix(name, ".go")
|
return !info.IsDir() && !strings.HasPrefix(name, ".") && strings.HasSuffix(name, ".go")
|
||||||
}, parser.ParseComments)
|
}, parser.ParseComments)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Error while parsing dir at '%s': %s", pkgpath, err)
|
beeLogger.Log.Fatalf("Error while parsing dir at '%s': %s", pkgpath, err)
|
||||||
}
|
}
|
||||||
for _, pkg := range astPkgs {
|
for _, pkg := range astPkgs {
|
||||||
for _, fl := range pkg.Files {
|
for _, fl := range pkg.Files {
|
||||||
@ -413,7 +417,7 @@ func isSystemPackage(pkgpath string) bool {
|
|||||||
goroot = runtime.GOROOT()
|
goroot = runtime.GOROOT()
|
||||||
}
|
}
|
||||||
if goroot == "" {
|
if goroot == "" {
|
||||||
logger.Fatalf("GOROOT environment variable is not set or empty")
|
beeLogger.Log.Fatalf("GOROOT environment variable is not set or empty")
|
||||||
}
|
}
|
||||||
|
|
||||||
wg, _ := filepath.EvalSymlinks(filepath.Join(goroot, "src", "pkg", pkgpath))
|
wg, _ := filepath.EvalSymlinks(filepath.Join(goroot, "src", "pkg", pkgpath))
|
||||||
@ -423,11 +427,7 @@ func isSystemPackage(pkgpath string) bool {
|
|||||||
|
|
||||||
//TODO(zh):support go1.4
|
//TODO(zh):support go1.4
|
||||||
wg, _ = filepath.EvalSymlinks(filepath.Join(goroot, "src", pkgpath))
|
wg, _ = filepath.EvalSymlinks(filepath.Join(goroot, "src", pkgpath))
|
||||||
if utils.FileExists(wg) {
|
return utils.FileExists(wg)
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
return false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func peekNextSplitString(ss string) (s string, spacePos int) {
|
func peekNextSplitString(ss string) (s string, spacePos int) {
|
||||||
@ -481,7 +481,7 @@ func parserComments(comments *ast.CommentGroup, funcName, controllerName, pkgpat
|
|||||||
ss = strings.TrimSpace(ss[pos:])
|
ss = strings.TrimSpace(ss[pos:])
|
||||||
schemaName, pos := peekNextSplitString(ss)
|
schemaName, pos := peekNextSplitString(ss)
|
||||||
if schemaName == "" {
|
if schemaName == "" {
|
||||||
logger.Fatalf("[%s.%s] Schema must follow {object} or {array}", controllerName, funcName)
|
beeLogger.Log.Fatalf("[%s.%s] Schema must follow {object} or {array}", controllerName, funcName)
|
||||||
}
|
}
|
||||||
if strings.HasPrefix(schemaName, "[]") {
|
if strings.HasPrefix(schemaName, "[]") {
|
||||||
schemaName = schemaName[2:]
|
schemaName = schemaName[2:]
|
||||||
@ -496,7 +496,7 @@ func parserComments(comments *ast.CommentGroup, funcName, controllerName, pkgpat
|
|||||||
m, mod, realTypes := getModel(schemaName)
|
m, mod, realTypes := getModel(schemaName)
|
||||||
schema.Ref = "#/definitions/" + m
|
schema.Ref = "#/definitions/" + m
|
||||||
if _, ok := modelsList[pkgpath+controllerName]; !ok {
|
if _, ok := modelsList[pkgpath+controllerName]; !ok {
|
||||||
modelsList[pkgpath+controllerName] = make(map[string]swagger.Schema, 0)
|
modelsList[pkgpath+controllerName] = make(map[string]swagger.Schema)
|
||||||
}
|
}
|
||||||
modelsList[pkgpath+controllerName][schemaName] = mod
|
modelsList[pkgpath+controllerName][schemaName] = mod
|
||||||
appendModels(pkgpath, controllerName, realTypes)
|
appendModels(pkgpath, controllerName, realTypes)
|
||||||
@ -518,7 +518,7 @@ func parserComments(comments *ast.CommentGroup, funcName, controllerName, pkgpat
|
|||||||
para := swagger.Parameter{}
|
para := swagger.Parameter{}
|
||||||
p := getparams(strings.TrimSpace(t[len("@Param "):]))
|
p := getparams(strings.TrimSpace(t[len("@Param "):]))
|
||||||
if len(p) < 4 {
|
if len(p) < 4 {
|
||||||
logger.Fatal(controllerName + "_" + funcName + "'s comments @Param should have at least 4 params")
|
beeLogger.Log.Fatal(controllerName + "_" + funcName + "'s comments @Param should have at least 4 params")
|
||||||
}
|
}
|
||||||
para.Name = p[0]
|
para.Name = p[0]
|
||||||
switch p[1] {
|
switch p[1] {
|
||||||
@ -533,7 +533,7 @@ func parserComments(comments *ast.CommentGroup, funcName, controllerName, pkgpat
|
|||||||
case "body":
|
case "body":
|
||||||
break
|
break
|
||||||
default:
|
default:
|
||||||
logger.Warnf("[%s.%s] Unknown param location: %s. Possible values are `query`, `header`, `path`, `formData` or `body`.\n", controllerName, funcName, p[1])
|
beeLogger.Log.Warnf("[%s.%s] Unknown param location: %s. Possible values are `query`, `header`, `path`, `formData` or `body`.\n", controllerName, funcName, p[1])
|
||||||
}
|
}
|
||||||
para.In = p[1]
|
para.In = p[1]
|
||||||
pp := strings.Split(p[2], ".")
|
pp := strings.Split(p[2], ".")
|
||||||
@ -544,7 +544,7 @@ func parserComments(comments *ast.CommentGroup, funcName, controllerName, pkgpat
|
|||||||
Ref: "#/definitions/" + m,
|
Ref: "#/definitions/" + m,
|
||||||
}
|
}
|
||||||
if _, ok := modelsList[pkgpath+controllerName]; !ok {
|
if _, ok := modelsList[pkgpath+controllerName]; !ok {
|
||||||
modelsList[pkgpath+controllerName] = make(map[string]swagger.Schema, 0)
|
modelsList[pkgpath+controllerName] = make(map[string]swagger.Schema)
|
||||||
}
|
}
|
||||||
modelsList[pkgpath+controllerName][typ] = mod
|
modelsList[pkgpath+controllerName][typ] = mod
|
||||||
appendModels(pkgpath, controllerName, realTypes)
|
appendModels(pkgpath, controllerName, realTypes)
|
||||||
@ -564,7 +564,7 @@ func parserComments(comments *ast.CommentGroup, funcName, controllerName, pkgpat
|
|||||||
paraType = typeFormat[0]
|
paraType = typeFormat[0]
|
||||||
paraFormat = typeFormat[1]
|
paraFormat = typeFormat[1]
|
||||||
} else {
|
} else {
|
||||||
logger.Warnf("[%s.%s] Unknown param type: %s\n", controllerName, funcName, typ)
|
beeLogger.Log.Warnf("[%s.%s] Unknown param type: %s\n", controllerName, funcName, typ)
|
||||||
}
|
}
|
||||||
if isArray {
|
if isArray {
|
||||||
para.Type = "array"
|
para.Type = "array"
|
||||||
@ -717,7 +717,7 @@ func getModel(str string) (objectname string, m swagger.Schema, realTypes []stri
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if m.Title == "" {
|
if m.Title == "" {
|
||||||
logger.Warnf("Cannot find the object: %s", str)
|
beeLogger.Log.Warnf("Cannot find the object: %s", str)
|
||||||
// TODO remove when all type have been supported
|
// TODO remove when all type have been supported
|
||||||
//os.Exit(1)
|
//os.Exit(1)
|
||||||
}
|
}
|
||||||
@ -732,7 +732,7 @@ func getModel(str string) (objectname string, m swagger.Schema, realTypes []stri
|
|||||||
func parseObject(d *ast.Object, k string, m *swagger.Schema, realTypes *[]string, astPkgs map[string]*ast.Package, packageName string) {
|
func parseObject(d *ast.Object, k string, m *swagger.Schema, realTypes *[]string, astPkgs map[string]*ast.Package, packageName string) {
|
||||||
ts, ok := d.Decl.(*ast.TypeSpec)
|
ts, ok := d.Decl.(*ast.TypeSpec)
|
||||||
if !ok {
|
if !ok {
|
||||||
logger.Fatalf("Unknown type without TypeSec: %v\n", d)
|
beeLogger.Log.Fatalf("Unknown type without TypeSec: %v\n", d)
|
||||||
}
|
}
|
||||||
// TODO support other types, such as `ArrayType`, `MapType`, `InterfaceType` etc...
|
// TODO support other types, such as `ArrayType`, `MapType`, `InterfaceType` etc...
|
||||||
st, ok := ts.Type.(*ast.StructType)
|
st, ok := ts.Type.(*ast.StructType)
|
||||||
@ -808,7 +808,7 @@ func parseObject(d *ast.Object, k string, m *swagger.Schema, realTypes *[]string
|
|||||||
mp.Default = str2RealType(res[1], realType)
|
mp.Default = str2RealType(res[1], realType)
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
logger.Warnf("Invalid default value: %s", defaultValue)
|
beeLogger.Log.Warnf("Invalid default value: %s", defaultValue)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -899,16 +899,6 @@ func isBasicType(Type string) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
// regexp get json tag
|
|
||||||
func grepJSONTag(tag string) string {
|
|
||||||
r, _ := regexp.Compile(`json:"([^"]*)"`)
|
|
||||||
matches := r.FindAllStringSubmatch(tag, -1)
|
|
||||||
if len(matches) > 0 {
|
|
||||||
return matches[0][1]
|
|
||||||
}
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
// append models
|
// append models
|
||||||
func appendModels(pkgpath, controllerName string, realTypes []string) {
|
func appendModels(pkgpath, controllerName string, realTypes []string) {
|
||||||
for _, realType := range realTypes {
|
for _, realType := range realTypes {
|
||||||
@ -956,7 +946,7 @@ func str2RealType(s string, typ string) interface{} {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Warnf("Invalid default value type '%s': %s", typ, s)
|
beeLogger.Log.Warnf("Invalid default value type '%s': %s", typ, s)
|
||||||
return s
|
return s
|
||||||
}
|
}
|
||||||
|
|
369
hproseapp.go
369
hproseapp.go
@ -1,369 +0,0 @@
|
|||||||
// Copyright 2013 bee 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 main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"os"
|
|
||||||
path "path/filepath"
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
var cmdHproseapp = &Command{
|
|
||||||
// CustomFlags: true,
|
|
||||||
UsageLine: "hprose [appname]",
|
|
||||||
Short: "Creates an RPC application based on Hprose and Beego frameworks",
|
|
||||||
Long: `
|
|
||||||
The command 'hprose' creates an RPC application based on both Beego and Hprose (http://hprose.com/).
|
|
||||||
|
|
||||||
{{"To scaffold out your application, use:"|bold}}
|
|
||||||
|
|
||||||
$ bee hprose [appname] [-tables=""] [-driver=mysql] [-conn=root:@tcp(127.0.0.1:3306)/test]
|
|
||||||
|
|
||||||
If 'conn' is empty, the command will generate a sample application. Otherwise the command
|
|
||||||
will connect to your database and generate models based on the existing tables.
|
|
||||||
|
|
||||||
The command 'hprose' creates a folder named [appname] with the following structure:
|
|
||||||
|
|
||||||
├── main.go
|
|
||||||
├── {{"conf"|foldername}}
|
|
||||||
│ └── app.conf
|
|
||||||
└── {{"models"|foldername}}
|
|
||||||
└── object.go
|
|
||||||
└── user.go
|
|
||||||
`,
|
|
||||||
PreRun: func(cmd *Command, args []string) { ShowShortVersionBanner() },
|
|
||||||
Run: createhprose,
|
|
||||||
}
|
|
||||||
|
|
||||||
var hproseconf = `appname = {{.Appname}}
|
|
||||||
httpport = 8080
|
|
||||||
runmode = dev
|
|
||||||
autorender = false
|
|
||||||
copyrequestbody = true
|
|
||||||
EnableDocs = true
|
|
||||||
`
|
|
||||||
var hproseMaingo = `package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"reflect"
|
|
||||||
|
|
||||||
"{{.Appname}}/models"
|
|
||||||
"github.com/hprose/hprose-golang/rpc"
|
|
||||||
|
|
||||||
"github.com/astaxie/beego"
|
|
||||||
)
|
|
||||||
|
|
||||||
func logInvokeHandler(
|
|
||||||
name string,
|
|
||||||
args []reflect.Value,
|
|
||||||
context rpc.Context,
|
|
||||||
next rpc.NextInvokeHandler) (results []reflect.Value, err error) {
|
|
||||||
fmt.Printf("%s(%v) = ", name, args)
|
|
||||||
results, err = next(name, args, context)
|
|
||||||
fmt.Printf("%v %v\r\n", results, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
// Create WebSocketServer
|
|
||||||
// service := rpc.NewWebSocketService()
|
|
||||||
|
|
||||||
// Create Http Server
|
|
||||||
service := rpc.NewHTTPService()
|
|
||||||
|
|
||||||
// Use Logger Middleware
|
|
||||||
service.AddInvokeHandler(logInvokeHandler)
|
|
||||||
|
|
||||||
// Publish Functions
|
|
||||||
service.AddFunction("AddOne", models.AddOne)
|
|
||||||
service.AddFunction("GetOne", models.GetOne)
|
|
||||||
|
|
||||||
// Start Service
|
|
||||||
beego.Handler("/", service)
|
|
||||||
beego.Run()
|
|
||||||
}
|
|
||||||
`
|
|
||||||
|
|
||||||
var hproseMainconngo = `package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"reflect"
|
|
||||||
|
|
||||||
"{{.Appname}}/models"
|
|
||||||
"github.com/hprose/hprose-golang/rpc"
|
|
||||||
|
|
||||||
"github.com/astaxie/beego"
|
|
||||||
"github.com/astaxie/beego/orm"
|
|
||||||
{{.DriverPkg}}
|
|
||||||
)
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
orm.RegisterDataBase("default", "{{.DriverName}}", "{{.conn}}")
|
|
||||||
}
|
|
||||||
|
|
||||||
func logInvokeHandler(
|
|
||||||
name string,
|
|
||||||
args []reflect.Value,
|
|
||||||
context rpc.Context,
|
|
||||||
next rpc.NextInvokeHandler) (results []reflect.Value, err error) {
|
|
||||||
fmt.Printf("%s(%v) = ", name, args)
|
|
||||||
results, err = next(name, args, context)
|
|
||||||
fmt.Printf("%v %v\r\n", results, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
// Create WebSocketServer
|
|
||||||
// service := rpc.NewWebSocketService()
|
|
||||||
|
|
||||||
// Create Http Server
|
|
||||||
service := rpc.NewHTTPService()
|
|
||||||
|
|
||||||
// Use Logger Middleware
|
|
||||||
service.AddInvokeHandler(logInvokeHandler)
|
|
||||||
|
|
||||||
{{HproseFunctionList}}
|
|
||||||
|
|
||||||
// Start Service
|
|
||||||
beego.Handler("/", service)
|
|
||||||
beego.Run()
|
|
||||||
}
|
|
||||||
|
|
||||||
`
|
|
||||||
|
|
||||||
var hproseModels = `package models
|
|
||||||
|
|
||||||
import (
|
|
||||||
"errors"
|
|
||||||
"strconv"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
Objects map[string]*Object
|
|
||||||
)
|
|
||||||
|
|
||||||
type Object struct {
|
|
||||||
ObjectId string
|
|
||||||
Score int64
|
|
||||||
PlayerName string
|
|
||||||
}
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
Objects = make(map[string]*Object)
|
|
||||||
Objects["hjkhsbnmn123"] = &Object{"hjkhsbnmn123", 100, "astaxie"}
|
|
||||||
Objects["mjjkxsxsaa23"] = &Object{"mjjkxsxsaa23", 101, "someone"}
|
|
||||||
}
|
|
||||||
|
|
||||||
func AddOne(object Object) (ObjectId string) {
|
|
||||||
object.ObjectId = "astaxie" + strconv.FormatInt(time.Now().UnixNano(), 10)
|
|
||||||
Objects[object.ObjectId] = &object
|
|
||||||
return object.ObjectId
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetOne(ObjectId string) (object *Object, err error) {
|
|
||||||
if v, ok := Objects[ObjectId]; ok {
|
|
||||||
return v, nil
|
|
||||||
}
|
|
||||||
return nil, errors.New("ObjectId Not Exist")
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetAll() map[string]*Object {
|
|
||||||
return Objects
|
|
||||||
}
|
|
||||||
|
|
||||||
func Update(ObjectId string, Score int64) (err error) {
|
|
||||||
if v, ok := Objects[ObjectId]; ok {
|
|
||||||
v.Score = Score
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return errors.New("ObjectId Not Exist")
|
|
||||||
}
|
|
||||||
|
|
||||||
func Delete(ObjectId string) {
|
|
||||||
delete(Objects, ObjectId)
|
|
||||||
}
|
|
||||||
|
|
||||||
`
|
|
||||||
|
|
||||||
var hproseModels2 = `package models
|
|
||||||
|
|
||||||
import (
|
|
||||||
"errors"
|
|
||||||
"strconv"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
UserList map[string]*User
|
|
||||||
)
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
UserList = make(map[string]*User)
|
|
||||||
u := User{"user_11111", "astaxie", "11111", Profile{"male", 20, "Singapore", "astaxie@gmail.com"}}
|
|
||||||
UserList["user_11111"] = &u
|
|
||||||
}
|
|
||||||
|
|
||||||
type User struct {
|
|
||||||
Id string
|
|
||||||
Username string
|
|
||||||
Password string
|
|
||||||
Profile Profile
|
|
||||||
}
|
|
||||||
|
|
||||||
type Profile struct {
|
|
||||||
Gender string
|
|
||||||
Age int
|
|
||||||
Address string
|
|
||||||
Email string
|
|
||||||
}
|
|
||||||
|
|
||||||
func AddUser(u User) string {
|
|
||||||
u.Id = "user_" + strconv.FormatInt(time.Now().UnixNano(), 10)
|
|
||||||
UserList[u.Id] = &u
|
|
||||||
return u.Id
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetUser(uid string) (u *User, err error) {
|
|
||||||
if u, ok := UserList[uid]; ok {
|
|
||||||
return u, nil
|
|
||||||
}
|
|
||||||
return nil, errors.New("User not exists")
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetAllUsers() map[string]*User {
|
|
||||||
return UserList
|
|
||||||
}
|
|
||||||
|
|
||||||
func UpdateUser(uid string, uu *User) (a *User, err error) {
|
|
||||||
if u, ok := UserList[uid]; ok {
|
|
||||||
if uu.Username != "" {
|
|
||||||
u.Username = uu.Username
|
|
||||||
}
|
|
||||||
if uu.Password != "" {
|
|
||||||
u.Password = uu.Password
|
|
||||||
}
|
|
||||||
if uu.Profile.Age != 0 {
|
|
||||||
u.Profile.Age = uu.Profile.Age
|
|
||||||
}
|
|
||||||
if uu.Profile.Address != "" {
|
|
||||||
u.Profile.Address = uu.Profile.Address
|
|
||||||
}
|
|
||||||
if uu.Profile.Gender != "" {
|
|
||||||
u.Profile.Gender = uu.Profile.Gender
|
|
||||||
}
|
|
||||||
if uu.Profile.Email != "" {
|
|
||||||
u.Profile.Email = uu.Profile.Email
|
|
||||||
}
|
|
||||||
return u, nil
|
|
||||||
}
|
|
||||||
return nil, errors.New("User Not Exist")
|
|
||||||
}
|
|
||||||
|
|
||||||
func Login(username, password string) bool {
|
|
||||||
for _, u := range UserList {
|
|
||||||
if u.Username == username && u.Password == password {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func DeleteUser(uid string) {
|
|
||||||
delete(UserList, uid)
|
|
||||||
}
|
|
||||||
`
|
|
||||||
|
|
||||||
var hproseAddFunctions = []string{}
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
cmdHproseapp.Flag.Var(&tables, "tables", "List of table names separated by a comma.")
|
|
||||||
cmdHproseapp.Flag.Var(&driver, "driver", "Database driver. Either mysql, postgres or sqlite.")
|
|
||||||
cmdHproseapp.Flag.Var(&conn, "conn", "Connection string used by the driver to connect to a database instance.")
|
|
||||||
}
|
|
||||||
|
|
||||||
func createhprose(cmd *Command, args []string) int {
|
|
||||||
output := cmd.Out()
|
|
||||||
|
|
||||||
if len(args) != 1 {
|
|
||||||
logger.Fatal("Argument [appname] is missing")
|
|
||||||
}
|
|
||||||
|
|
||||||
curpath, _ := os.Getwd()
|
|
||||||
if len(args) > 1 {
|
|
||||||
cmd.Flag.Parse(args[1:])
|
|
||||||
}
|
|
||||||
apppath, packpath, err := checkEnv(args[0])
|
|
||||||
if err != nil {
|
|
||||||
logger.Fatalf("%s", err)
|
|
||||||
}
|
|
||||||
if driver == "" {
|
|
||||||
driver = "mysql"
|
|
||||||
}
|
|
||||||
if conn == "" {
|
|
||||||
}
|
|
||||||
|
|
||||||
logger.Info("Creating Hprose application...")
|
|
||||||
|
|
||||||
os.MkdirAll(apppath, 0755)
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", apppath, "\x1b[0m")
|
|
||||||
os.Mkdir(path.Join(apppath, "conf"), 0755)
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "conf"), "\x1b[0m")
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "conf", "app.conf"), "\x1b[0m")
|
|
||||||
WriteToFile(path.Join(apppath, "conf", "app.conf"),
|
|
||||||
strings.Replace(hproseconf, "{{.Appname}}", args[0], -1))
|
|
||||||
|
|
||||||
if conn != "" {
|
|
||||||
logger.Infof("Using '%s' as 'driver'", driver)
|
|
||||||
logger.Infof("Using '%s' as 'conn'", conn)
|
|
||||||
logger.Infof("Using '%s' as 'tables'", tables)
|
|
||||||
generateHproseAppcode(string(driver), string(conn), "1", string(tables), path.Join(curpath, args[0]))
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "main.go"), "\x1b[0m")
|
|
||||||
maingoContent := strings.Replace(hproseMainconngo, "{{.Appname}}", packpath, -1)
|
|
||||||
maingoContent = strings.Replace(maingoContent, "{{.DriverName}}", string(driver), -1)
|
|
||||||
maingoContent = strings.Replace(maingoContent, "{{HproseFunctionList}}", strings.Join(hproseAddFunctions, ""), -1)
|
|
||||||
if driver == "mysql" {
|
|
||||||
maingoContent = strings.Replace(maingoContent, "{{.DriverPkg}}", `_ "github.com/go-sql-driver/mysql"`, -1)
|
|
||||||
} else if driver == "postgres" {
|
|
||||||
maingoContent = strings.Replace(maingoContent, "{{.DriverPkg}}", `_ "github.com/lib/pq"`, -1)
|
|
||||||
}
|
|
||||||
WriteToFile(path.Join(apppath, "main.go"),
|
|
||||||
strings.Replace(
|
|
||||||
maingoContent,
|
|
||||||
"{{.conn}}",
|
|
||||||
conn.String(),
|
|
||||||
-1,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
os.Mkdir(path.Join(apppath, "models"), 0755)
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "models"), "\x1b[0m")
|
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "models", "object.go"), "\x1b[0m")
|
|
||||||
WriteToFile(path.Join(apppath, "models", "object.go"), apiModels)
|
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "models", "user.go"), "\x1b[0m")
|
|
||||||
WriteToFile(path.Join(apppath, "models", "user.go"), apiModels2)
|
|
||||||
|
|
||||||
fmt.Fprintf(output, "\t%s%screate%s\t %s%s\n", "\x1b[32m", "\x1b[1m", "\x1b[21m", path.Join(apppath, "main.go"), "\x1b[0m")
|
|
||||||
WriteToFile(path.Join(apppath, "main.go"),
|
|
||||||
strings.Replace(hproseMaingo, "{{.Appname}}", packpath, -1))
|
|
||||||
}
|
|
||||||
logger.Success("New Hprose application successfully created!")
|
|
||||||
return 0
|
|
||||||
}
|
|
@ -12,7 +12,7 @@
|
|||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
|
||||||
package main
|
package colors
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
@ -53,7 +53,7 @@ func NewModeColorWriter(w io.Writer, mode outputMode) io.Writer {
|
|||||||
return w
|
return w
|
||||||
}
|
}
|
||||||
|
|
||||||
func bold(message string) string {
|
func Bold(message string) string {
|
||||||
return fmt.Sprintf("\x1b[1m%s\x1b[21m", message)
|
return fmt.Sprintf("\x1b[1m%s\x1b[21m", message)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -102,47 +102,47 @@ func Magenta(message string) string {
|
|||||||
return fmt.Sprintf("\x1b[35m%s\x1b[0m", message)
|
return fmt.Sprintf("\x1b[35m%s\x1b[0m", message)
|
||||||
}
|
}
|
||||||
|
|
||||||
// BlackBold returns a black bold string
|
// BlackBold returns a black Bold string
|
||||||
func BlackBold(message string) string {
|
func BlackBold(message string) string {
|
||||||
return fmt.Sprintf("\x1b[30m%s\x1b[0m", bold(message))
|
return fmt.Sprintf("\x1b[30m%s\x1b[0m", Bold(message))
|
||||||
}
|
}
|
||||||
|
|
||||||
// WhiteBold returns a white bold string
|
// WhiteBold returns a white Bold string
|
||||||
func WhiteBold(message string) string {
|
func WhiteBold(message string) string {
|
||||||
return fmt.Sprintf("\x1b[37m%s\x1b[0m", bold(message))
|
return fmt.Sprintf("\x1b[37m%s\x1b[0m", Bold(message))
|
||||||
}
|
}
|
||||||
|
|
||||||
// CyanBold returns a cyan bold string
|
// CyanBold returns a cyan Bold string
|
||||||
func CyanBold(message string) string {
|
func CyanBold(message string) string {
|
||||||
return fmt.Sprintf("\x1b[36m%s\x1b[0m", bold(message))
|
return fmt.Sprintf("\x1b[36m%s\x1b[0m", Bold(message))
|
||||||
}
|
}
|
||||||
|
|
||||||
// BlueBold returns a blue bold string
|
// BlueBold returns a blue Bold string
|
||||||
func BlueBold(message string) string {
|
func BlueBold(message string) string {
|
||||||
return fmt.Sprintf("\x1b[34m%s\x1b[0m", bold(message))
|
return fmt.Sprintf("\x1b[34m%s\x1b[0m", Bold(message))
|
||||||
}
|
}
|
||||||
|
|
||||||
// RedBold returns a red bold string
|
// RedBold returns a red Bold string
|
||||||
func RedBold(message string) string {
|
func RedBold(message string) string {
|
||||||
return fmt.Sprintf("\x1b[31m%s\x1b[0m", bold(message))
|
return fmt.Sprintf("\x1b[31m%s\x1b[0m", Bold(message))
|
||||||
}
|
}
|
||||||
|
|
||||||
// GreenBold returns a green bold string
|
// GreenBold returns a green Bold string
|
||||||
func GreenBold(message string) string {
|
func GreenBold(message string) string {
|
||||||
return fmt.Sprintf("\x1b[32m%s\x1b[0m", bold(message))
|
return fmt.Sprintf("\x1b[32m%s\x1b[0m", Bold(message))
|
||||||
}
|
}
|
||||||
|
|
||||||
// YellowBold returns a yellow bold string
|
// YellowBold returns a yellow Bold string
|
||||||
func YellowBold(message string) string {
|
func YellowBold(message string) string {
|
||||||
return fmt.Sprintf("\x1b[33m%s\x1b[0m", bold(message))
|
return fmt.Sprintf("\x1b[33m%s\x1b[0m", Bold(message))
|
||||||
}
|
}
|
||||||
|
|
||||||
// GrayBold returns a gray bold string
|
// GrayBold returns a gray Bold string
|
||||||
func GrayBold(message string) string {
|
func GrayBold(message string) string {
|
||||||
return fmt.Sprintf("\x1b[37m%s\x1b[0m", bold(message))
|
return fmt.Sprintf("\x1b[37m%s\x1b[0m", Bold(message))
|
||||||
}
|
}
|
||||||
|
|
||||||
// MagentaBold returns a magenta bold string
|
// MagentaBold returns a magenta Bold string
|
||||||
func MagentaBold(message string) string {
|
func MagentaBold(message string) string {
|
||||||
return fmt.Sprintf("\x1b[35m%s\x1b[0m", bold(message))
|
return fmt.Sprintf("\x1b[35m%s\x1b[0m", Bold(message))
|
||||||
}
|
}
|
@ -14,7 +14,7 @@
|
|||||||
|
|
||||||
// +build !windows
|
// +build !windows
|
||||||
|
|
||||||
package main
|
package colors
|
||||||
|
|
||||||
import "io"
|
import "io"
|
||||||
|
|
@ -14,7 +14,7 @@
|
|||||||
|
|
||||||
// +build windows
|
// +build windows
|
||||||
|
|
||||||
package main
|
package colors
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
@ -11,7 +11,7 @@
|
|||||||
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
package main
|
package beeLogger
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
@ -22,19 +22,22 @@ import (
|
|||||||
"sync"
|
"sync"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
"text/template"
|
"text/template"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/beego/bee/logger/colors"
|
||||||
)
|
)
|
||||||
|
|
||||||
var errInvalidLogLevel = errors.New("logger: invalid log level")
|
var errInvalidLogLevel = errors.New("logger: invalid log level")
|
||||||
|
|
||||||
const (
|
const (
|
||||||
levelCritical = iota
|
levelDebug = iota
|
||||||
levelFatal
|
|
||||||
levelSuccess
|
|
||||||
levelHint
|
|
||||||
levelDebug
|
|
||||||
levelInfo
|
|
||||||
levelWarn
|
|
||||||
levelError
|
levelError
|
||||||
|
levelFatal
|
||||||
|
levelCritical
|
||||||
|
levelSuccess
|
||||||
|
levelWarn
|
||||||
|
levelInfo
|
||||||
|
levelHint
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
@ -42,6 +45,9 @@ var (
|
|||||||
instance *BeeLogger
|
instance *BeeLogger
|
||||||
once sync.Once
|
once sync.Once
|
||||||
)
|
)
|
||||||
|
var debugMode = os.Getenv("DEBUG_ENABLED") == "1"
|
||||||
|
|
||||||
|
var logLevel = levelInfo
|
||||||
|
|
||||||
// BeeLogger logs logging records to the specified io.Writer
|
// BeeLogger logs logging records to the specified io.Writer
|
||||||
type BeeLogger struct {
|
type BeeLogger struct {
|
||||||
@ -59,6 +65,8 @@ type LogRecord struct {
|
|||||||
LineNo int
|
LineNo int
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var Log = GetBeeLogger(os.Stdout)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
logRecordTemplate *template.Template
|
logRecordTemplate *template.Template
|
||||||
debugLogRecordTemplate *template.Template
|
debugLogRecordTemplate *template.Template
|
||||||
@ -80,11 +88,15 @@ func GetBeeLogger(w io.Writer) *BeeLogger {
|
|||||||
"EndLine": EndLine,
|
"EndLine": EndLine,
|
||||||
}
|
}
|
||||||
logRecordTemplate, err = template.New("simpleLogFormat").Funcs(funcs).Parse(simpleLogFormat)
|
logRecordTemplate, err = template.New("simpleLogFormat").Funcs(funcs).Parse(simpleLogFormat)
|
||||||
MustCheck(err)
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
debugLogRecordTemplate, err = template.New("debugLogFormat").Funcs(funcs).Parse(debugLogFormat)
|
debugLogRecordTemplate, err = template.New("debugLogFormat").Funcs(funcs).Parse(debugLogFormat)
|
||||||
MustCheck(err)
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
instance = &BeeLogger{output: NewColorWriter(w)}
|
instance = &BeeLogger{output: colors.NewColorWriter(w)}
|
||||||
})
|
})
|
||||||
return instance
|
return instance
|
||||||
}
|
}
|
||||||
@ -93,7 +105,17 @@ func GetBeeLogger(w io.Writer) *BeeLogger {
|
|||||||
func (l *BeeLogger) SetOutput(w io.Writer) {
|
func (l *BeeLogger) SetOutput(w io.Writer) {
|
||||||
l.mu.Lock()
|
l.mu.Lock()
|
||||||
defer l.mu.Unlock()
|
defer l.mu.Unlock()
|
||||||
l.output = NewColorWriter(w)
|
l.output = colors.NewColorWriter(w)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now returns the current local time in the specified layout
|
||||||
|
func Now(layout string) string {
|
||||||
|
return time.Now().Format(layout)
|
||||||
|
}
|
||||||
|
|
||||||
|
// EndLine returns the a newline escape character
|
||||||
|
func EndLine() string {
|
||||||
|
return "\n"
|
||||||
}
|
}
|
||||||
|
|
||||||
func (l *BeeLogger) getLevelTag(level int) string {
|
func (l *BeeLogger) getLevelTag(level int) string {
|
||||||
@ -122,21 +144,21 @@ func (l *BeeLogger) getLevelTag(level int) string {
|
|||||||
func (l *BeeLogger) getColorLevel(level int) string {
|
func (l *BeeLogger) getColorLevel(level int) string {
|
||||||
switch level {
|
switch level {
|
||||||
case levelCritical:
|
case levelCritical:
|
||||||
return RedBold(l.getLevelTag(level))
|
return colors.RedBold(l.getLevelTag(level))
|
||||||
case levelFatal:
|
case levelFatal:
|
||||||
return RedBold(l.getLevelTag(level))
|
return colors.RedBold(l.getLevelTag(level))
|
||||||
case levelInfo:
|
case levelInfo:
|
||||||
return BlueBold(l.getLevelTag(level))
|
return colors.BlueBold(l.getLevelTag(level))
|
||||||
case levelHint:
|
case levelHint:
|
||||||
return CyanBold(l.getLevelTag(level))
|
return colors.CyanBold(l.getLevelTag(level))
|
||||||
case levelDebug:
|
case levelDebug:
|
||||||
return YellowBold(l.getLevelTag(level))
|
return colors.YellowBold(l.getLevelTag(level))
|
||||||
case levelError:
|
case levelError:
|
||||||
return RedBold(l.getLevelTag(level))
|
return colors.RedBold(l.getLevelTag(level))
|
||||||
case levelWarn:
|
case levelWarn:
|
||||||
return YellowBold(l.getLevelTag(level))
|
return colors.YellowBold(l.getLevelTag(level))
|
||||||
case levelSuccess:
|
case levelSuccess:
|
||||||
return GreenBold(l.getLevelTag(level))
|
return colors.GreenBold(l.getLevelTag(level))
|
||||||
default:
|
default:
|
||||||
panic(errInvalidLogLevel)
|
panic(errInvalidLogLevel)
|
||||||
}
|
}
|
||||||
@ -145,6 +167,9 @@ func (l *BeeLogger) getColorLevel(level int) string {
|
|||||||
// mustLog logs the message according to the specified level and arguments.
|
// mustLog logs the message according to the specified level and arguments.
|
||||||
// It panics in case of an error.
|
// It panics in case of an error.
|
||||||
func (l *BeeLogger) mustLog(level int, message string, args ...interface{}) {
|
func (l *BeeLogger) mustLog(level int, message string, args ...interface{}) {
|
||||||
|
if level > logLevel {
|
||||||
|
return
|
||||||
|
}
|
||||||
// Acquire the lock
|
// Acquire the lock
|
||||||
l.mu.Lock()
|
l.mu.Lock()
|
||||||
defer l.mu.Unlock()
|
defer l.mu.Unlock()
|
||||||
@ -157,13 +182,15 @@ func (l *BeeLogger) mustLog(level int, message string, args ...interface{}) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
err := logRecordTemplate.Execute(l.output, record)
|
err := logRecordTemplate.Execute(l.output, record)
|
||||||
MustCheck(err)
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// mustLogDebug logs a debug message only if debug mode
|
// mustLogDebug logs a debug message only if debug mode
|
||||||
// is enabled. i.e. DEBUG_ENABLED="1"
|
// is enabled. i.e. DEBUG_ENABLED="1"
|
||||||
func (l *BeeLogger) mustLogDebug(message string, file string, line int, args ...interface{}) {
|
func (l *BeeLogger) mustLogDebug(message string, file string, line int, args ...interface{}) {
|
||||||
if !IsDebugEnabled() {
|
if !debugMode {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -179,7 +206,9 @@ func (l *BeeLogger) mustLogDebug(message string, file string, line int, args ...
|
|||||||
Filename: filepath.Base(file),
|
Filename: filepath.Base(file),
|
||||||
}
|
}
|
||||||
err := debugLogRecordTemplate.Execute(l.output, record)
|
err := debugLogRecordTemplate.Execute(l.output, record)
|
||||||
MustCheck(err)
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Debug outputs a debug log message
|
// Debug outputs a debug log message
|
77
main.go
Normal file
77
main.go
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
// Copyright 2013 bee 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 main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"flag"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/beego/bee/cmd"
|
||||||
|
"github.com/beego/bee/cmd/commands"
|
||||||
|
"github.com/beego/bee/config"
|
||||||
|
"github.com/beego/bee/generate/swaggergen"
|
||||||
|
"github.com/beego/bee/utils"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
currentpath, _ := os.Getwd()
|
||||||
|
|
||||||
|
flag.Usage = cmd.Usage
|
||||||
|
flag.Parse()
|
||||||
|
log.SetFlags(0)
|
||||||
|
|
||||||
|
args := flag.Args()
|
||||||
|
|
||||||
|
if len(args) < 1 {
|
||||||
|
cmd.Usage()
|
||||||
|
os.Exit(2)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if args[0] == "help" {
|
||||||
|
cmd.Help(args[1:])
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, c := range commands.AvailableCommands {
|
||||||
|
if c.Name() == args[0] && c.Run != nil {
|
||||||
|
c.Flag.Usage = func() { c.Usage() }
|
||||||
|
if c.CustomFlags {
|
||||||
|
args = args[1:]
|
||||||
|
} else {
|
||||||
|
c.Flag.Parse(args[1:])
|
||||||
|
args = c.Flag.Args()
|
||||||
|
}
|
||||||
|
|
||||||
|
if c.PreRun != nil {
|
||||||
|
c.PreRun(c, args)
|
||||||
|
}
|
||||||
|
|
||||||
|
config.LoadConfig()
|
||||||
|
|
||||||
|
// Check if current directory is inside the GOPATH,
|
||||||
|
// if so parse the packages inside it.
|
||||||
|
if strings.Contains(currentpath, utils.GetGOPATHs()[0]+"/src") && cmd.IfGenerateDocs(c.Name(), args) {
|
||||||
|
swaggergen.ParsePackagesFromDir(currentpath)
|
||||||
|
}
|
||||||
|
|
||||||
|
os.Exit(c.Run(c, args))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
utils.PrintErrorAndExit("Unknown subcommand", cmd.ErrorTemplate)
|
||||||
|
}
|
155
rundocs.go
155
rundocs.go
@ -1,155 +0,0 @@
|
|||||||
// Copyright 2013 bee 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 main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"archive/zip"
|
|
||||||
"fmt"
|
|
||||||
"io"
|
|
||||||
"net/http"
|
|
||||||
"os"
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
var cmdRundocs = &Command{
|
|
||||||
UsageLine: "rundocs [-isDownload=true] [-docport=8888]",
|
|
||||||
Short: "rundocs will run the docs server,default is 8089",
|
|
||||||
Long: `
|
|
||||||
-d meaning will download the docs file from github
|
|
||||||
-p meaning server the Server on which port, default is 8089
|
|
||||||
|
|
||||||
`,
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
|
||||||
swaggerVersion = "2"
|
|
||||||
swaggerlink = "https://github.com/beego/swagger/archive/v" + swaggerVersion + ".zip"
|
|
||||||
)
|
|
||||||
|
|
||||||
type docValue string
|
|
||||||
|
|
||||||
func (d *docValue) String() string {
|
|
||||||
return fmt.Sprint(*d)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *docValue) Set(value string) error {
|
|
||||||
*d = docValue(value)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
var isDownload docValue
|
|
||||||
var docport docValue
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
cmdRundocs.Run = runDocs
|
|
||||||
cmdRundocs.PreRun = func(cmd *Command, args []string) { ShowShortVersionBanner() }
|
|
||||||
cmdRundocs.Flag.Var(&isDownload, "isDownload", "weather download the Swagger Docs")
|
|
||||||
cmdRundocs.Flag.Var(&docport, "docport", "doc server port")
|
|
||||||
}
|
|
||||||
|
|
||||||
func runDocs(cmd *Command, args []string) int {
|
|
||||||
if isDownload == "true" {
|
|
||||||
downloadFromURL(swaggerlink, "swagger.zip")
|
|
||||||
err := unzipAndDelete("swagger.zip")
|
|
||||||
if err != nil {
|
|
||||||
logger.Errorf("Error while unzipping 'swagger.zip' file: %s", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if docport == "" {
|
|
||||||
docport = "8089"
|
|
||||||
}
|
|
||||||
if _, err := os.Stat("swagger"); err != nil && os.IsNotExist(err) {
|
|
||||||
logger.Fatal("No Swagger dist found. Run: bee rundocs -isDownload=true")
|
|
||||||
}
|
|
||||||
|
|
||||||
logger.Infof("Starting the docs server on: http://127.0.0.1:%s", docport)
|
|
||||||
|
|
||||||
err := http.ListenAndServe(":"+string(docport), http.FileServer(http.Dir("swagger")))
|
|
||||||
if err != nil {
|
|
||||||
logger.Fatalf("%s", err)
|
|
||||||
}
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
func downloadFromURL(url, fileName string) {
|
|
||||||
var down bool
|
|
||||||
if fd, err := os.Stat(fileName); err != nil && os.IsNotExist(err) {
|
|
||||||
down = true
|
|
||||||
} else if fd.Size() == int64(0) {
|
|
||||||
down = true
|
|
||||||
} else {
|
|
||||||
logger.Infof("'%s' already exists", fileName)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if down {
|
|
||||||
logger.Infof("Downloading '%s' to '%s'...", url, fileName)
|
|
||||||
output, err := os.Create(fileName)
|
|
||||||
if err != nil {
|
|
||||||
logger.Errorf("Error while creating '%s': %s", fileName, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer output.Close()
|
|
||||||
|
|
||||||
response, err := http.Get(url)
|
|
||||||
if err != nil {
|
|
||||||
logger.Errorf("Error while downloading '%s': %s", url, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer response.Body.Close()
|
|
||||||
|
|
||||||
n, err := io.Copy(output, response.Body)
|
|
||||||
if err != nil {
|
|
||||||
logger.Errorf("Error while downloading '%s': %s", url, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
logger.Successf("%d bytes downloaded!", n)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func unzipAndDelete(src string) error {
|
|
||||||
logger.Infof("Unzipping '%s'...", src)
|
|
||||||
r, err := zip.OpenReader(src)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
defer r.Close()
|
|
||||||
|
|
||||||
rp := strings.NewReplacer("swagger-"+swaggerVersion, "swagger")
|
|
||||||
for _, f := range r.File {
|
|
||||||
rc, err := f.Open()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
defer rc.Close()
|
|
||||||
|
|
||||||
fname := rp.Replace(f.Name)
|
|
||||||
if f.FileInfo().IsDir() {
|
|
||||||
os.MkdirAll(fname, f.Mode())
|
|
||||||
} else {
|
|
||||||
f, err := os.OpenFile(
|
|
||||||
fname, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, f.Mode())
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
defer f.Close()
|
|
||||||
|
|
||||||
_, err = io.Copy(f, rc)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
logger.Successf("Done! Deleting '%s'...", src)
|
|
||||||
return os.RemoveAll(src)
|
|
||||||
}
|
|
102
test.go
102
test.go
@ -1,102 +0,0 @@
|
|||||||
// Copyright 2013 bee 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 main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"os"
|
|
||||||
"os/exec"
|
|
||||||
path "path/filepath"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
_ "github.com/smartystreets/goconvey/convey"
|
|
||||||
)
|
|
||||||
|
|
||||||
var cmdTest = &Command{
|
|
||||||
UsageLine: "test [appname]",
|
|
||||||
Short: "test the app",
|
|
||||||
Long: ``,
|
|
||||||
}
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
cmdTest.Run = testApp
|
|
||||||
cmdTest.PreRun = func(cmd *Command, args []string) { ShowShortVersionBanner() }
|
|
||||||
}
|
|
||||||
|
|
||||||
func safePathAppend(arr []string, paths ...string) []string {
|
|
||||||
for _, path := range paths {
|
|
||||||
if pathExists(path) {
|
|
||||||
arr = append(arr, path)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return arr
|
|
||||||
}
|
|
||||||
|
|
||||||
func pathExists(path string) bool {
|
|
||||||
_, err := os.Stat(path)
|
|
||||||
return err == nil || os.IsExist(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
var started = make(chan bool)
|
|
||||||
|
|
||||||
func testApp(cmd *Command, args []string) int {
|
|
||||||
if len(args) != 1 {
|
|
||||||
logger.Fatalf("Failed to start: %s", "argument 'appname' is missing")
|
|
||||||
}
|
|
||||||
|
|
||||||
currpath, _ := os.Getwd()
|
|
||||||
|
|
||||||
logger.Debugf("Current path: %s", __FILE__(), __LINE__(), currpath)
|
|
||||||
|
|
||||||
err := loadConfig()
|
|
||||||
if err != nil {
|
|
||||||
logger.Fatalf("Failed to load configuration: %s", err)
|
|
||||||
}
|
|
||||||
var paths []string
|
|
||||||
readAppDirectories(currpath, &paths)
|
|
||||||
|
|
||||||
NewWatcher(paths, nil, false)
|
|
||||||
appname = args[0]
|
|
||||||
for {
|
|
||||||
select {
|
|
||||||
case <-started:
|
|
||||||
runTest()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func runTest() {
|
|
||||||
logger.Info("Start testing...")
|
|
||||||
time.Sleep(time.Second * 1)
|
|
||||||
crupwd, _ := os.Getwd()
|
|
||||||
testDir := path.Join(crupwd, "tests")
|
|
||||||
if pathExists(testDir) {
|
|
||||||
os.Chdir(testDir)
|
|
||||||
}
|
|
||||||
|
|
||||||
var err error
|
|
||||||
icmd := exec.Command("go", "test")
|
|
||||||
icmd.Stdout = os.Stdout
|
|
||||||
icmd.Stderr = os.Stderr
|
|
||||||
logger.Info("============== Test Begin ===================")
|
|
||||||
err = icmd.Run()
|
|
||||||
logger.Info("============== Test End =====================")
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
logger.Error("============== Test failed ===================")
|
|
||||||
logger.Errorf("Cause: %s", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
logger.Success("Test Completed")
|
|
||||||
}
|
|
28
testdata/router/router.go
vendored
28
testdata/router/router.go
vendored
@ -1,28 +0,0 @@
|
|||||||
package router
|
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/astaxie/beego"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Router struct {
|
|
||||||
beego.Controller
|
|
||||||
}
|
|
||||||
|
|
||||||
func (r *Router) Get() {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func (r *Router) Post() {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
type Controller struct {
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Controller) Put() {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Controller) Delete() {
|
|
||||||
|
|
||||||
}
|
|
14
utils/doc_value.go
Normal file
14
utils/doc_value.go
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
package utils
|
||||||
|
|
||||||
|
import "fmt"
|
||||||
|
|
||||||
|
type DocValue string
|
||||||
|
|
||||||
|
func (d *DocValue) String() string {
|
||||||
|
return fmt.Sprint(*d)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *DocValue) Set(value string) error {
|
||||||
|
*d = DocValue(value)
|
||||||
|
return nil
|
||||||
|
}
|
14
utils/list_opts.go
Normal file
14
utils/list_opts.go
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
package utils
|
||||||
|
|
||||||
|
import "fmt"
|
||||||
|
|
||||||
|
type ListOpts []string
|
||||||
|
|
||||||
|
func (opts *ListOpts) String() string {
|
||||||
|
return fmt.Sprint(*opts)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (opts *ListOpts) Set(value string) error {
|
||||||
|
*opts = append(*opts, value)
|
||||||
|
return nil
|
||||||
|
}
|
84
utils/notification.go
Normal file
84
utils/notification.go
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
// Copyright 2017 bee 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 utils
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os/exec"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"runtime"
|
||||||
|
|
||||||
|
"github.com/beego/bee/config"
|
||||||
|
)
|
||||||
|
|
||||||
|
const appName = "Beego"
|
||||||
|
|
||||||
|
func Notify(text, title string) {
|
||||||
|
if !config.Conf.EnableNotification {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
switch runtime.GOOS {
|
||||||
|
case "darwin":
|
||||||
|
osxNotify(text, title)
|
||||||
|
case "linux":
|
||||||
|
linuxNotify(text, title)
|
||||||
|
case "windows":
|
||||||
|
windowsNotify(text, title)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func osxNotify(text, title string) {
|
||||||
|
var cmd *exec.Cmd
|
||||||
|
if existTerminalNotifier() {
|
||||||
|
cmd = exec.Command("terminal-notifier", "-title", appName, "-message", text, "-subtitle", title)
|
||||||
|
} else if MacOSVersionSupport() {
|
||||||
|
notification := fmt.Sprintf("display notification \"%s\" with title \"%s\" subtitle \"%s\"", text, appName, title)
|
||||||
|
cmd = exec.Command("osascript", "-e", notification)
|
||||||
|
} else {
|
||||||
|
cmd = exec.Command("growlnotify", "-n", appName, "-m", title)
|
||||||
|
}
|
||||||
|
cmd.Run()
|
||||||
|
}
|
||||||
|
|
||||||
|
func windowsNotify(text, title string) {
|
||||||
|
exec.Command("growlnotify", "/i:", "", "/t:", title, text).Run()
|
||||||
|
}
|
||||||
|
|
||||||
|
func linuxNotify(text, title string) {
|
||||||
|
exec.Command("notify-send", "-i", "", title, text).Run()
|
||||||
|
}
|
||||||
|
|
||||||
|
func existTerminalNotifier() bool {
|
||||||
|
cmd := exec.Command("which", "terminal-notifier")
|
||||||
|
err := cmd.Start()
|
||||||
|
if err != nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
err = cmd.Wait()
|
||||||
|
return err != nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func MacOSVersionSupport() bool {
|
||||||
|
cmd := exec.Command("sw_vers", "-productVersion")
|
||||||
|
check, _ := cmd.Output()
|
||||||
|
version := strings.Split(string(check), ".")
|
||||||
|
major, _ := strconv.Atoi(version[0])
|
||||||
|
minor, _ := strconv.Atoi(version[1])
|
||||||
|
if major < 10 || (major == 10 && minor < 9) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
15
utils/str_flag.go
Normal file
15
utils/str_flag.go
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
package utils
|
||||||
|
|
||||||
|
import "fmt"
|
||||||
|
|
||||||
|
// The string flag list, implemented flag.Value interface
|
||||||
|
type StrFlags []string
|
||||||
|
|
||||||
|
func (s *StrFlags) String() string {
|
||||||
|
return fmt.Sprintf("%s", *s)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *StrFlags) Set(value string) error {
|
||||||
|
*s = append(*s, value)
|
||||||
|
return nil
|
||||||
|
}
|
@ -12,7 +12,7 @@
|
|||||||
// License for the specific language governing permissions and limitations
|
// License for the specific language governing permissions and limitations
|
||||||
// under the License.
|
// under the License.
|
||||||
|
|
||||||
package main
|
package utils
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
@ -26,7 +26,10 @@ import (
|
|||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
"text/template"
|
"text/template"
|
||||||
"time"
|
"unicode"
|
||||||
|
|
||||||
|
beeLogger "github.com/beego/bee/logger"
|
||||||
|
"github.com/beego/bee/logger/colors"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Go is a basic promise implementation: it wraps calls a function in a goroutine
|
// Go is a basic promise implementation: it wraps calls a function in a goroutine
|
||||||
@ -39,18 +42,8 @@ func Go(f func() error) chan error {
|
|||||||
return ch
|
return ch
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now returns the current local time in the specified layout
|
|
||||||
func Now(layout string) string {
|
|
||||||
return time.Now().Format(layout)
|
|
||||||
}
|
|
||||||
|
|
||||||
// EndLine returns the a newline escape character
|
|
||||||
func EndLine() string {
|
|
||||||
return "\n"
|
|
||||||
}
|
|
||||||
|
|
||||||
// IsExist returns whether a file or directory exists.
|
// IsExist returns whether a file or directory exists.
|
||||||
func isExist(path string) bool {
|
func IsExist(path string) bool {
|
||||||
_, err := os.Stat(path)
|
_, err := os.Stat(path)
|
||||||
return err == nil || os.IsExist(err)
|
return err == nil || os.IsExist(err)
|
||||||
}
|
}
|
||||||
@ -107,7 +100,7 @@ func IsBeegoProject(thePath string) bool {
|
|||||||
}()
|
}()
|
||||||
|
|
||||||
if err := <-c; err != nil {
|
if err := <-c; err != nil {
|
||||||
logger.Fatalf("Unable to walk '%s' tree: %s", thePath, err)
|
beeLogger.Log.Fatalf("Unable to walk '%s' tree: %s", thePath, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(mainFiles) > 0 {
|
if len(mainFiles) > 0 {
|
||||||
@ -121,7 +114,7 @@ func IsBeegoProject(thePath string) bool {
|
|||||||
func SearchGOPATHs(app string) (bool, string, string) {
|
func SearchGOPATHs(app string) (bool, string, string) {
|
||||||
gps := GetGOPATHs()
|
gps := GetGOPATHs()
|
||||||
if len(gps) == 0 {
|
if len(gps) == 0 {
|
||||||
logger.Fatal("GOPATH environment variable is not set or empty")
|
beeLogger.Log.Fatal("GOPATH environment variable is not set or empty")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Lookup the application inside the user workspace(s)
|
// Lookup the application inside the user workspace(s)
|
||||||
@ -135,7 +128,7 @@ func SearchGOPATHs(app string) (bool, string, string) {
|
|||||||
currentPath = app
|
currentPath = app
|
||||||
}
|
}
|
||||||
|
|
||||||
if isExist(currentPath) {
|
if IsExist(currentPath) {
|
||||||
return true, gopath, currentPath
|
return true, gopath, currentPath
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -147,11 +140,11 @@ func SearchGOPATHs(app string) (bool, string, string) {
|
|||||||
// confirmations. If the input is not recognized, it will ask again. The function does not return
|
// confirmations. If the input is not recognized, it will ask again. The function does not return
|
||||||
// until it gets a valid response from the user. Typically, you should use fmt to print out a question
|
// until it gets a valid response from the user. Typically, you should use fmt to print out a question
|
||||||
// before calling askForConfirmation. E.g. fmt.Println("WARNING: Are you sure? (yes/no)")
|
// before calling askForConfirmation. E.g. fmt.Println("WARNING: Are you sure? (yes/no)")
|
||||||
func askForConfirmation() bool {
|
func AskForConfirmation() bool {
|
||||||
var response string
|
var response string
|
||||||
_, err := fmt.Scanln(&response)
|
_, err := fmt.Scanln(&response)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("%s", err)
|
beeLogger.Log.Fatalf("%s", err)
|
||||||
}
|
}
|
||||||
okayResponses := []string{"y", "Y", "yes", "Yes", "YES"}
|
okayResponses := []string{"y", "Y", "yes", "Yes", "YES"}
|
||||||
nokayResponses := []string{"n", "N", "no", "No", "NO"}
|
nokayResponses := []string{"n", "N", "no", "No", "NO"}
|
||||||
@ -161,7 +154,7 @@ func askForConfirmation() bool {
|
|||||||
return false
|
return false
|
||||||
} else {
|
} else {
|
||||||
fmt.Println("Please type yes or no and then press enter:")
|
fmt.Println("Please type yes or no and then press enter:")
|
||||||
return askForConfirmation()
|
return AskForConfirmation()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -175,7 +168,7 @@ func containsString(slice []string, element string) bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// snake string, XxYy to xx_yy
|
// snake string, XxYy to xx_yy
|
||||||
func snakeString(s string) string {
|
func SnakeString(s string) string {
|
||||||
data := make([]byte, 0, len(s)*2)
|
data := make([]byte, 0, len(s)*2)
|
||||||
j := false
|
j := false
|
||||||
num := len(s)
|
num := len(s)
|
||||||
@ -192,17 +185,17 @@ func snakeString(s string) string {
|
|||||||
return strings.ToLower(string(data[:]))
|
return strings.ToLower(string(data[:]))
|
||||||
}
|
}
|
||||||
|
|
||||||
func camelString(s string) string {
|
func CamelString(s string) string {
|
||||||
data := make([]byte, 0, len(s))
|
data := make([]byte, 0, len(s))
|
||||||
j := false
|
j := false
|
||||||
k := false
|
k := false
|
||||||
num := len(s) - 1
|
num := len(s) - 1
|
||||||
for i := 0; i <= num; i++ {
|
for i := 0; i <= num; i++ {
|
||||||
d := s[i]
|
d := s[i]
|
||||||
if k == false && d >= 'A' && d <= 'Z' {
|
if !k && d >= 'A' && d <= 'Z' {
|
||||||
k = true
|
k = true
|
||||||
}
|
}
|
||||||
if d >= 'a' && d <= 'z' && (j || k == false) {
|
if d >= 'a' && d <= 'z' && (j || !k) {
|
||||||
d = d - 32
|
d = d - 32
|
||||||
j = false
|
j = false
|
||||||
k = true
|
k = true
|
||||||
@ -218,7 +211,7 @@ func camelString(s string) string {
|
|||||||
|
|
||||||
// camelCase converts a _ delimited string to camel case
|
// camelCase converts a _ delimited string to camel case
|
||||||
// e.g. very_important_person => VeryImportantPerson
|
// e.g. very_important_person => VeryImportantPerson
|
||||||
func camelCase(in string) string {
|
func CamelCase(in string) string {
|
||||||
tokens := strings.Split(in, "_")
|
tokens := strings.Split(in, "_")
|
||||||
for i := range tokens {
|
for i := range tokens {
|
||||||
tokens[i] = strings.Title(strings.Trim(tokens[i], " "))
|
tokens[i] = strings.Title(strings.Trim(tokens[i], " "))
|
||||||
@ -227,25 +220,13 @@ func camelCase(in string) string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// formatSourceCode formats source files
|
// formatSourceCode formats source files
|
||||||
func formatSourceCode(filename string) {
|
func FormatSourceCode(filename string) {
|
||||||
cmd := exec.Command("gofmt", "-w", filename)
|
cmd := exec.Command("gofmt", "-w", filename)
|
||||||
if err := cmd.Run(); err != nil {
|
if err := cmd.Run(); err != nil {
|
||||||
logger.Warnf("Error while running gofmt: %s", err)
|
beeLogger.Log.Warnf("Error while running gofmt: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// The string flag list, implemented flag.Value interface
|
|
||||||
type strFlags []string
|
|
||||||
|
|
||||||
func (s *strFlags) String() string {
|
|
||||||
return fmt.Sprintf("%s", *s)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *strFlags) Set(value string) error {
|
|
||||||
*s = append(*s, value)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// CloseFile attempts to close the passed file
|
// CloseFile attempts to close the passed file
|
||||||
// or panics with the actual error
|
// or panics with the actual error
|
||||||
func CloseFile(f *os.File) {
|
func CloseFile(f *os.File) {
|
||||||
@ -260,11 +241,6 @@ func MustCheck(err error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func exitPrint(con string) {
|
|
||||||
fmt.Fprintln(os.Stderr, con)
|
|
||||||
os.Exit(2)
|
|
||||||
}
|
|
||||||
|
|
||||||
// WriteToFile creates a file and writes content to it
|
// WriteToFile creates a file and writes content to it
|
||||||
func WriteToFile(filename, content string) {
|
func WriteToFile(filename, content string) {
|
||||||
f, err := os.Create(filename)
|
f, err := os.Create(filename)
|
||||||
@ -274,20 +250,14 @@ func WriteToFile(filename, content string) {
|
|||||||
MustCheck(err)
|
MustCheck(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// IsDebugEnabled checks if DEBUG_ENABLED is set or not
|
|
||||||
func IsDebugEnabled() bool {
|
|
||||||
debugMode := os.Getenv("DEBUG_ENABLED")
|
|
||||||
return map[string]bool{"1": true, "0": false}[debugMode]
|
|
||||||
}
|
|
||||||
|
|
||||||
// __FILE__ returns the file name in which the function was invoked
|
// __FILE__ returns the file name in which the function was invoked
|
||||||
func __FILE__() string {
|
func FILE() string {
|
||||||
_, file, _, _ := runtime.Caller(1)
|
_, file, _, _ := runtime.Caller(1)
|
||||||
return file
|
return file
|
||||||
}
|
}
|
||||||
|
|
||||||
// __LINE__ returns the line number at which the function was invoked
|
// __LINE__ returns the line number at which the function was invoked
|
||||||
func __LINE__() int {
|
func LINE() int {
|
||||||
_, _, line, _ := runtime.Caller(1)
|
_, _, line, _ := runtime.Caller(1)
|
||||||
return line
|
return line
|
||||||
}
|
}
|
||||||
@ -296,9 +266,9 @@ func __LINE__() int {
|
|||||||
func BeeFuncMap() template.FuncMap {
|
func BeeFuncMap() template.FuncMap {
|
||||||
return template.FuncMap{
|
return template.FuncMap{
|
||||||
"trim": strings.TrimSpace,
|
"trim": strings.TrimSpace,
|
||||||
"bold": bold,
|
"bold": colors.Bold,
|
||||||
"headline": MagentaBold,
|
"headline": colors.MagentaBold,
|
||||||
"foldername": RedBold,
|
"foldername": colors.RedBold,
|
||||||
"endline": EndLine,
|
"endline": EndLine,
|
||||||
"tmpltostr": TmplToString,
|
"tmpltostr": TmplToString,
|
||||||
}
|
}
|
||||||
@ -315,3 +285,125 @@ func TmplToString(tmpl string, data interface{}) string {
|
|||||||
|
|
||||||
return doc.String()
|
return doc.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// EndLine returns the a newline escape character
|
||||||
|
func EndLine() string {
|
||||||
|
return "\n"
|
||||||
|
}
|
||||||
|
|
||||||
|
func Tmpl(text string, data interface{}) {
|
||||||
|
output := colors.NewColorWriter(os.Stderr)
|
||||||
|
|
||||||
|
t := template.New("Usage").Funcs(BeeFuncMap())
|
||||||
|
template.Must(t.Parse(text))
|
||||||
|
|
||||||
|
err := t.Execute(output, data)
|
||||||
|
if err != nil {
|
||||||
|
beeLogger.Log.Error(err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func CheckEnv(appname string) (apppath, packpath string, err error) {
|
||||||
|
gps := GetGOPATHs()
|
||||||
|
if len(gps) == 0 {
|
||||||
|
beeLogger.Log.Fatal("GOPATH environment variable is not set or empty")
|
||||||
|
}
|
||||||
|
currpath, _ := os.Getwd()
|
||||||
|
currpath = path.Join(currpath, appname)
|
||||||
|
for _, gpath := range gps {
|
||||||
|
gsrcpath := path.Join(gpath, "src")
|
||||||
|
if strings.HasPrefix(currpath, gsrcpath) {
|
||||||
|
packpath = strings.Replace(currpath[len(gsrcpath)+1:], string(filepath.Separator), "/", -1)
|
||||||
|
return currpath, packpath, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// In case of multiple paths in the GOPATH, by default
|
||||||
|
// we use the first path
|
||||||
|
gopath := gps[0]
|
||||||
|
|
||||||
|
beeLogger.Log.Warn("You current workdir is not inside $GOPATH/src.")
|
||||||
|
beeLogger.Log.Debugf("GOPATH: %s", FILE(), LINE(), gopath)
|
||||||
|
|
||||||
|
gosrcpath := path.Join(gopath, "src")
|
||||||
|
apppath = path.Join(gosrcpath, appname)
|
||||||
|
|
||||||
|
if _, e := os.Stat(apppath); !os.IsNotExist(e) {
|
||||||
|
err = fmt.Errorf("Cannot create application without removing '%s' first", apppath)
|
||||||
|
beeLogger.Log.Errorf("Path '%s' already exists", apppath)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
packpath = strings.Join(strings.Split(apppath[len(gosrcpath)+1:], string(filepath.Separator)), "/")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func PrintErrorAndExit(message, errorTemplate string) {
|
||||||
|
Tmpl(fmt.Sprintf(errorTemplate, message), nil)
|
||||||
|
os.Exit(2)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GoCommand executes the passed command using Go tool
|
||||||
|
func GoCommand(command string, args ...string) error {
|
||||||
|
allargs := []string{command}
|
||||||
|
allargs = append(allargs, args...)
|
||||||
|
goBuild := exec.Command("go", allargs...)
|
||||||
|
goBuild.Stderr = os.Stderr
|
||||||
|
return goBuild.Run()
|
||||||
|
}
|
||||||
|
|
||||||
|
// SplitQuotedFields is like strings.Fields but ignores spaces
|
||||||
|
// inside areas surrounded by single quotes.
|
||||||
|
// To specify a single quote use backslash to escape it: '\''
|
||||||
|
func SplitQuotedFields(in string) []string {
|
||||||
|
type stateEnum int
|
||||||
|
const (
|
||||||
|
inSpace stateEnum = iota
|
||||||
|
inField
|
||||||
|
inQuote
|
||||||
|
inQuoteEscaped
|
||||||
|
)
|
||||||
|
state := inSpace
|
||||||
|
r := []string{}
|
||||||
|
var buf bytes.Buffer
|
||||||
|
|
||||||
|
for _, ch := range in {
|
||||||
|
switch state {
|
||||||
|
case inSpace:
|
||||||
|
if ch == '\'' {
|
||||||
|
state = inQuote
|
||||||
|
} else if !unicode.IsSpace(ch) {
|
||||||
|
buf.WriteRune(ch)
|
||||||
|
state = inField
|
||||||
|
}
|
||||||
|
|
||||||
|
case inField:
|
||||||
|
if ch == '\'' {
|
||||||
|
state = inQuote
|
||||||
|
} else if unicode.IsSpace(ch) {
|
||||||
|
r = append(r, buf.String())
|
||||||
|
buf.Reset()
|
||||||
|
} else {
|
||||||
|
buf.WriteRune(ch)
|
||||||
|
}
|
||||||
|
|
||||||
|
case inQuote:
|
||||||
|
if ch == '\'' {
|
||||||
|
state = inField
|
||||||
|
} else if ch == '\\' {
|
||||||
|
state = inQuoteEscaped
|
||||||
|
} else {
|
||||||
|
buf.WriteRune(ch)
|
||||||
|
}
|
||||||
|
|
||||||
|
case inQuoteEscaped:
|
||||||
|
buf.WriteRune(ch)
|
||||||
|
state = inQuote
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if buf.Len() != 0 {
|
||||||
|
r = append(r, buf.String())
|
||||||
|
}
|
||||||
|
|
||||||
|
return r
|
||||||
|
}
|
13
vendor/github.com/astaxie/beego/utils/safemap.go
generated
vendored
13
vendor/github.com/astaxie/beego/utils/safemap.go
generated
vendored
@ -61,10 +61,8 @@ func (m *BeeMap) Set(k interface{}, v interface{}) bool {
|
|||||||
func (m *BeeMap) Check(k interface{}) bool {
|
func (m *BeeMap) Check(k interface{}) bool {
|
||||||
m.lock.RLock()
|
m.lock.RLock()
|
||||||
defer m.lock.RUnlock()
|
defer m.lock.RUnlock()
|
||||||
if _, ok := m.bm[k]; !ok {
|
_, ok := m.bm[k]
|
||||||
return false
|
return ok
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delete the given key and value.
|
// Delete the given key and value.
|
||||||
@ -84,3 +82,10 @@ func (m *BeeMap) Items() map[interface{}]interface{} {
|
|||||||
}
|
}
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Count returns the number of items within the map.
|
||||||
|
func (m *BeeMap) Count() int {
|
||||||
|
m.lock.RLock()
|
||||||
|
defer m.lock.RUnlock()
|
||||||
|
return len(m.bm)
|
||||||
|
}
|
||||||
|
4
vendor/github.com/jtolds/gls/LICENSE → vendor/github.com/derekparker/delve/LICENSE
generated
vendored
4
vendor/github.com/jtolds/gls/LICENSE → vendor/github.com/derekparker/delve/LICENSE
generated
vendored
@ -1,4 +1,6 @@
|
|||||||
Copyright (c) 2013, Space Monkey, Inc.
|
The MIT License (MIT)
|
||||||
|
|
||||||
|
Copyright (c) 2014 Derek Parker
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||||
this software and associated documentation files (the "Software"), to deal in
|
this software and associated documentation files (the "Software"), to deal in
|
134
vendor/github.com/derekparker/delve/pkg/config/config.go
generated
vendored
Normal file
134
vendor/github.com/derekparker/delve/pkg/config/config.go
generated
vendored
Normal file
@ -0,0 +1,134 @@
|
|||||||
|
package config
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"os/user"
|
||||||
|
"path"
|
||||||
|
|
||||||
|
yaml "gopkg.in/yaml.v2"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
configDir string = ".dlv"
|
||||||
|
configFile string = "config.yml"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Describes a rule for substitution of path to source code file.
|
||||||
|
type SubstitutePathRule struct {
|
||||||
|
// Directory path will be substituted if it matches `From`.
|
||||||
|
From string
|
||||||
|
// Path to which substitution is performed.
|
||||||
|
To string
|
||||||
|
}
|
||||||
|
|
||||||
|
// Slice of source code path substitution rules.
|
||||||
|
type SubstitutePathRules []SubstitutePathRule
|
||||||
|
|
||||||
|
// Config defines all configuration options available to be set through the config file.
|
||||||
|
type Config struct {
|
||||||
|
// Commands aliases.
|
||||||
|
Aliases map[string][]string
|
||||||
|
// Source code path substitution rules.
|
||||||
|
SubstitutePath SubstitutePathRules `yaml:"substitute-path"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// LoadConfig attempts to populate a Config object from the config.yml file.
|
||||||
|
func LoadConfig() *Config {
|
||||||
|
err := createConfigPath()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("Could not create config directory: %v.", err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
fullConfigFile, err := GetConfigFilePath(configFile)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("Unable to get config file path: %v.", err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
f, err := os.Open(fullConfigFile)
|
||||||
|
if err != nil {
|
||||||
|
createDefaultConfig(fullConfigFile)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
err := f.Close()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("Closing config file failed: %v.", err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
data, err := ioutil.ReadAll(f)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("Unable to read config data: %v.", err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var c Config
|
||||||
|
err = yaml.Unmarshal(data, &c)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("Unable to decode config file: %v.", err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return &c
|
||||||
|
}
|
||||||
|
|
||||||
|
func createDefaultConfig(path string) {
|
||||||
|
f, err := os.Create(path)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("Unable to create config file: %v.", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
err := f.Close()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("Closing config file failed: %v.", err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
err = writeDefaultConfig(f)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("Unable to write default configuration: %v.", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func writeDefaultConfig(f *os.File) error {
|
||||||
|
_, err := f.WriteString(
|
||||||
|
`# Configuration file for the delve debugger.
|
||||||
|
|
||||||
|
# This is the default configuration file. Available options are provided, but disabled.
|
||||||
|
# Delete the leading hash mark to enable an item.
|
||||||
|
|
||||||
|
# Provided aliases will be added to the default aliases for a given command.
|
||||||
|
aliases:
|
||||||
|
# command: ["alias1", "alias2"]
|
||||||
|
|
||||||
|
# Define sources path substitution rules. Can be used to rewrite a source path stored
|
||||||
|
# in program's debug information, if the sources were moved to a different place
|
||||||
|
# between compilation and debugging.
|
||||||
|
# Note that substitution rules will not be used for paths passed to "break" and "trace"
|
||||||
|
# commands.
|
||||||
|
substitute-path:
|
||||||
|
# - {from: path, to: path}
|
||||||
|
`)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// createConfigPath creates the directory structure at which all config files are saved.
|
||||||
|
func createConfigPath() error {
|
||||||
|
path, err := GetConfigFilePath("")
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return os.MkdirAll(path, 0700)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetConfigFilePath gets the full path to the given config file name.
|
||||||
|
func GetConfigFilePath(file string) (string, error) {
|
||||||
|
usr, err := user.Current()
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return path.Join(usr.HomeDir, configDir, file), nil
|
||||||
|
}
|
95
vendor/github.com/derekparker/delve/pkg/dwarf/frame/entries.go
generated
vendored
Normal file
95
vendor/github.com/derekparker/delve/pkg/dwarf/frame/entries.go
generated
vendored
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
package frame
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/binary"
|
||||||
|
"fmt"
|
||||||
|
"sort"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Represents a Common Information Entry in
|
||||||
|
// the Dwarf .debug_frame section.
|
||||||
|
type CommonInformationEntry struct {
|
||||||
|
Length uint32
|
||||||
|
CIE_id uint32
|
||||||
|
Version uint8
|
||||||
|
Augmentation string
|
||||||
|
CodeAlignmentFactor uint64
|
||||||
|
DataAlignmentFactor int64
|
||||||
|
ReturnAddressRegister uint64
|
||||||
|
InitialInstructions []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
// Represents a Frame Descriptor Entry in the
|
||||||
|
// Dwarf .debug_frame section.
|
||||||
|
type FrameDescriptionEntry struct {
|
||||||
|
Length uint32
|
||||||
|
CIE *CommonInformationEntry
|
||||||
|
Instructions []byte
|
||||||
|
begin, end uint64
|
||||||
|
order binary.ByteOrder
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns whether or not the given address is within the
|
||||||
|
// bounds of this frame.
|
||||||
|
func (fde *FrameDescriptionEntry) Cover(addr uint64) bool {
|
||||||
|
if (addr - fde.begin) < fde.end {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Address of first location for this frame.
|
||||||
|
func (fde *FrameDescriptionEntry) Begin() uint64 {
|
||||||
|
return fde.begin
|
||||||
|
}
|
||||||
|
|
||||||
|
// Address of last location for this frame.
|
||||||
|
func (fde *FrameDescriptionEntry) End() uint64 {
|
||||||
|
return fde.begin + fde.end
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set up frame for the given PC.
|
||||||
|
func (fde *FrameDescriptionEntry) EstablishFrame(pc uint64) *FrameContext {
|
||||||
|
return executeDwarfProgramUntilPC(fde, pc)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return the offset from the current SP that the return address is stored at.
|
||||||
|
func (fde *FrameDescriptionEntry) ReturnAddressOffset(pc uint64) (frameOffset, returnAddressOffset int64) {
|
||||||
|
frame := fde.EstablishFrame(pc)
|
||||||
|
return frame.cfa.offset, frame.regs[fde.CIE.ReturnAddressRegister].offset
|
||||||
|
}
|
||||||
|
|
||||||
|
type FrameDescriptionEntries []*FrameDescriptionEntry
|
||||||
|
|
||||||
|
func NewFrameIndex() FrameDescriptionEntries {
|
||||||
|
return make(FrameDescriptionEntries, 0, 1000)
|
||||||
|
}
|
||||||
|
|
||||||
|
type NoFDEForPCError struct {
|
||||||
|
PC uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
func (err *NoFDEForPCError) Error() string {
|
||||||
|
return fmt.Sprintf("could not find FDE for PC %#v", err.PC)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns the Frame Description Entry for the given PC.
|
||||||
|
func (fdes FrameDescriptionEntries) FDEForPC(pc uint64) (*FrameDescriptionEntry, error) {
|
||||||
|
idx := sort.Search(len(fdes), func(i int) bool {
|
||||||
|
if fdes[i].Cover(pc) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if fdes[i].LessThan(pc) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
if idx == len(fdes) {
|
||||||
|
return nil, &NoFDEForPCError{pc}
|
||||||
|
}
|
||||||
|
return fdes[idx], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (frame *FrameDescriptionEntry) LessThan(pc uint64) bool {
|
||||||
|
return frame.End() <= pc
|
||||||
|
}
|
164
vendor/github.com/derekparker/delve/pkg/dwarf/frame/expression_constants.go
generated
vendored
Normal file
164
vendor/github.com/derekparker/delve/pkg/dwarf/frame/expression_constants.go
generated
vendored
Normal file
@ -0,0 +1,164 @@
|
|||||||
|
package frame
|
||||||
|
|
||||||
|
// Operation opcodes
|
||||||
|
const (
|
||||||
|
DW_OP_addr = 0x03
|
||||||
|
DW_OP_const1s = 0x09
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
DW_OP_const2u = 0x0a
|
||||||
|
DW_OP_const2s = 0x0b
|
||||||
|
DW_OP_const4u = iota
|
||||||
|
DW_OP_const4s
|
||||||
|
DW_OP_const8u
|
||||||
|
DW_OP_const8s
|
||||||
|
DW_OP_constu
|
||||||
|
DW_OP_consts
|
||||||
|
DW_OP_dup
|
||||||
|
DW_OP_drop
|
||||||
|
DW_OP_over
|
||||||
|
DW_OP_pick
|
||||||
|
DW_OP_swap
|
||||||
|
DW_OP_rot
|
||||||
|
DW_OP_xderef
|
||||||
|
DW_OP_abs
|
||||||
|
DW_OP_and
|
||||||
|
DW_OP_div
|
||||||
|
DW_OP_minus
|
||||||
|
DW_OP_mod
|
||||||
|
DW_OP_mul
|
||||||
|
DW_OP_neg
|
||||||
|
DW_OP_not
|
||||||
|
DW_OP_or
|
||||||
|
DW_OP_plus
|
||||||
|
DW_OP_plus_uconst
|
||||||
|
DW_OP_shl
|
||||||
|
DW_OP_shr
|
||||||
|
DW_OP_shra
|
||||||
|
DW_OP_xor
|
||||||
|
DW_OP_skip
|
||||||
|
DW_OP_bra
|
||||||
|
DW_OP_eq
|
||||||
|
DW_OP_ge
|
||||||
|
DW_OP_gt
|
||||||
|
DW_OP_le
|
||||||
|
DW_OP_lt
|
||||||
|
DW_OP_ne
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
DW_OP_lit0 = 0x30
|
||||||
|
DW_OP_lit1 = 0x31
|
||||||
|
DW_OP_lit2 = iota
|
||||||
|
DW_OP_lit3
|
||||||
|
DW_OP_lit4
|
||||||
|
DW_OP_lit5
|
||||||
|
DW_OP_lit6
|
||||||
|
DW_OP_lit7
|
||||||
|
DW_OP_lit8
|
||||||
|
DW_OP_lit9
|
||||||
|
DW_OP_lit10
|
||||||
|
DW_OP_lit11
|
||||||
|
DW_OP_lit12
|
||||||
|
DW_OP_lit13
|
||||||
|
DW_OP_lit14
|
||||||
|
DW_OP_lit15
|
||||||
|
DW_OP_lit16
|
||||||
|
DW_OP_lit17
|
||||||
|
DW_OP_lit18
|
||||||
|
DW_OP_lit19
|
||||||
|
DW_OP_lit20
|
||||||
|
DW_OP_lit21
|
||||||
|
DW_OP_lit22
|
||||||
|
DW_OP_lit23
|
||||||
|
DW_OP_lit24
|
||||||
|
DW_OP_lit25
|
||||||
|
DW_OP_lit26
|
||||||
|
DW_OP_lit27
|
||||||
|
DW_OP_lit28
|
||||||
|
DW_OP_lit29
|
||||||
|
DW_OP_lit30
|
||||||
|
DW_OP_lit31
|
||||||
|
DW_OP_reg0
|
||||||
|
DW_OP_reg1
|
||||||
|
DW_OP_reg2
|
||||||
|
DW_OP_reg3
|
||||||
|
DW_OP_reg4
|
||||||
|
DW_OP_reg5
|
||||||
|
DW_OP_reg6
|
||||||
|
DW_OP_reg7
|
||||||
|
DW_OP_reg8
|
||||||
|
DW_OP_reg9
|
||||||
|
DW_OP_reg10
|
||||||
|
DW_OP_reg11
|
||||||
|
DW_OP_reg12
|
||||||
|
DW_OP_reg13
|
||||||
|
DW_OP_reg14
|
||||||
|
DW_OP_reg15
|
||||||
|
DW_OP_reg16
|
||||||
|
DW_OP_reg17
|
||||||
|
DW_OP_reg18
|
||||||
|
DW_OP_reg19
|
||||||
|
DW_OP_reg20
|
||||||
|
DW_OP_reg21
|
||||||
|
DW_OP_reg22
|
||||||
|
DW_OP_reg23
|
||||||
|
DW_OP_reg24
|
||||||
|
DW_OP_reg25
|
||||||
|
DW_OP_reg26
|
||||||
|
DW_OP_reg27
|
||||||
|
DW_OP_reg28
|
||||||
|
DW_OP_reg29
|
||||||
|
DW_OP_reg30
|
||||||
|
DW_OP_reg31
|
||||||
|
DW_OP_breg0
|
||||||
|
DW_OP_breg1
|
||||||
|
DW_OP_breg2
|
||||||
|
DW_OP_breg3
|
||||||
|
DW_OP_breg4
|
||||||
|
DW_OP_breg5
|
||||||
|
DW_OP_breg6
|
||||||
|
DW_OP_breg7
|
||||||
|
DW_OP_breg8
|
||||||
|
DW_OP_breg9
|
||||||
|
DW_OP_breg10
|
||||||
|
DW_OP_breg11
|
||||||
|
DW_OP_breg12
|
||||||
|
DW_OP_breg13
|
||||||
|
DW_OP_breg14
|
||||||
|
DW_OP_breg15
|
||||||
|
DW_OP_breg16
|
||||||
|
DW_OP_breg17
|
||||||
|
DW_OP_breg18
|
||||||
|
DW_OP_breg19
|
||||||
|
DW_OP_breg20
|
||||||
|
DW_OP_breg21
|
||||||
|
DW_OP_breg22
|
||||||
|
DW_OP_breg23
|
||||||
|
DW_OP_breg24
|
||||||
|
DW_OP_breg25
|
||||||
|
DW_OP_breg26
|
||||||
|
DW_OP_breg27
|
||||||
|
DW_OP_breg28
|
||||||
|
DW_OP_breg29
|
||||||
|
DW_OP_breg30
|
||||||
|
DW_OP_breg31
|
||||||
|
DW_OP_regx
|
||||||
|
DW_OP_fbreg
|
||||||
|
DW_OP_bregx
|
||||||
|
DW_OP_piece
|
||||||
|
DW_OP_deref_size
|
||||||
|
DW_OP_xderef_size
|
||||||
|
DW_OP_nop
|
||||||
|
DW_OP_push_object_address
|
||||||
|
DW_OP_call2
|
||||||
|
DW_OP_call4
|
||||||
|
DW_OP_call_ref
|
||||||
|
DW_OP_form_tls_address
|
||||||
|
DW_OP_call_frame_cfa
|
||||||
|
DW_OP_bit_piece
|
||||||
|
|
||||||
|
DW_OP_lo_user = 0xe0
|
||||||
|
DW_OP_hi_user = 0xff
|
||||||
|
)
|
125
vendor/github.com/derekparker/delve/pkg/dwarf/frame/parser.go
generated
vendored
Normal file
125
vendor/github.com/derekparker/delve/pkg/dwarf/frame/parser.go
generated
vendored
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
// Package frame contains data structures and
|
||||||
|
// related functions for parsing and searching
|
||||||
|
// through Dwarf .debug_frame data.
|
||||||
|
package frame
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/binary"
|
||||||
|
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/util"
|
||||||
|
)
|
||||||
|
|
||||||
|
type parsefunc func(*parseContext) parsefunc
|
||||||
|
|
||||||
|
type parseContext struct {
|
||||||
|
buf *bytes.Buffer
|
||||||
|
entries FrameDescriptionEntries
|
||||||
|
common *CommonInformationEntry
|
||||||
|
frame *FrameDescriptionEntry
|
||||||
|
length uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse takes in data (a byte slice) and returns a slice of
|
||||||
|
// commonInformationEntry structures. Each commonInformationEntry
|
||||||
|
// has a slice of frameDescriptionEntry structures.
|
||||||
|
func Parse(data []byte, order binary.ByteOrder) FrameDescriptionEntries {
|
||||||
|
var (
|
||||||
|
buf = bytes.NewBuffer(data)
|
||||||
|
pctx = &parseContext{buf: buf, entries: NewFrameIndex()}
|
||||||
|
)
|
||||||
|
|
||||||
|
for fn := parselength; buf.Len() != 0; {
|
||||||
|
fn = fn(pctx)
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := range pctx.entries {
|
||||||
|
pctx.entries[i].order = order
|
||||||
|
}
|
||||||
|
|
||||||
|
return pctx.entries
|
||||||
|
}
|
||||||
|
|
||||||
|
func cieEntry(data []byte) bool {
|
||||||
|
return bytes.Equal(data, []byte{0xff, 0xff, 0xff, 0xff})
|
||||||
|
}
|
||||||
|
|
||||||
|
func parselength(ctx *parseContext) parsefunc {
|
||||||
|
var data = ctx.buf.Next(8)
|
||||||
|
|
||||||
|
ctx.length = binary.LittleEndian.Uint32(data[:4]) - 4 // take off the length of the CIE id / CIE pointer.
|
||||||
|
|
||||||
|
if cieEntry(data[4:]) {
|
||||||
|
ctx.common = &CommonInformationEntry{Length: ctx.length}
|
||||||
|
return parseCIE
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx.frame = &FrameDescriptionEntry{Length: ctx.length, CIE: ctx.common}
|
||||||
|
return parseFDE
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseFDE(ctx *parseContext) parsefunc {
|
||||||
|
r := ctx.buf.Next(int(ctx.length))
|
||||||
|
|
||||||
|
ctx.frame.begin = binary.LittleEndian.Uint64(r[:8])
|
||||||
|
ctx.frame.end = binary.LittleEndian.Uint64(r[8:16])
|
||||||
|
|
||||||
|
// Insert into the tree after setting address range begin
|
||||||
|
// otherwise compares won't work.
|
||||||
|
ctx.entries = append(ctx.entries, ctx.frame)
|
||||||
|
|
||||||
|
// The rest of this entry consists of the instructions
|
||||||
|
// so we can just grab all of the data from the buffer
|
||||||
|
// cursor to length.
|
||||||
|
ctx.frame.Instructions = r[16:]
|
||||||
|
ctx.length = 0
|
||||||
|
|
||||||
|
return parselength
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseCIE(ctx *parseContext) parsefunc {
|
||||||
|
data := ctx.buf.Next(int(ctx.length))
|
||||||
|
buf := bytes.NewBuffer(data)
|
||||||
|
// parse version
|
||||||
|
ctx.common.Version = data[0]
|
||||||
|
|
||||||
|
// parse augmentation
|
||||||
|
ctx.common.Augmentation, _ = util.ParseString(buf)
|
||||||
|
|
||||||
|
// parse code alignment factor
|
||||||
|
ctx.common.CodeAlignmentFactor, _ = util.DecodeULEB128(buf)
|
||||||
|
|
||||||
|
// parse data alignment factor
|
||||||
|
ctx.common.DataAlignmentFactor, _ = util.DecodeSLEB128(buf)
|
||||||
|
|
||||||
|
// parse return address register
|
||||||
|
ctx.common.ReturnAddressRegister, _ = util.DecodeULEB128(buf)
|
||||||
|
|
||||||
|
// parse initial instructions
|
||||||
|
// The rest of this entry consists of the instructions
|
||||||
|
// so we can just grab all of the data from the buffer
|
||||||
|
// cursor to length.
|
||||||
|
ctx.common.InitialInstructions = buf.Bytes() //ctx.buf.Next(int(ctx.length))
|
||||||
|
ctx.length = 0
|
||||||
|
|
||||||
|
return parselength
|
||||||
|
}
|
||||||
|
|
||||||
|
// DwarfEndian determines the endianness of the DWARF by using the version number field in the debug_info section
|
||||||
|
// Trick borrowed from "debug/dwarf".New()
|
||||||
|
func DwarfEndian(infoSec []byte) binary.ByteOrder {
|
||||||
|
if len(infoSec) < 6 {
|
||||||
|
return binary.BigEndian
|
||||||
|
}
|
||||||
|
x, y := infoSec[4], infoSec[5]
|
||||||
|
switch {
|
||||||
|
case x == 0 && y == 0:
|
||||||
|
return binary.BigEndian
|
||||||
|
case x == 0:
|
||||||
|
return binary.BigEndian
|
||||||
|
case y == 0:
|
||||||
|
return binary.LittleEndian
|
||||||
|
default:
|
||||||
|
return binary.BigEndian
|
||||||
|
}
|
||||||
|
}
|
429
vendor/github.com/derekparker/delve/pkg/dwarf/frame/table.go
generated
vendored
Normal file
429
vendor/github.com/derekparker/delve/pkg/dwarf/frame/table.go
generated
vendored
Normal file
@ -0,0 +1,429 @@
|
|||||||
|
package frame
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/binary"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/util"
|
||||||
|
)
|
||||||
|
|
||||||
|
type CurrentFrameAddress struct {
|
||||||
|
register uint64
|
||||||
|
offset int64
|
||||||
|
expression []byte
|
||||||
|
rule byte
|
||||||
|
}
|
||||||
|
|
||||||
|
type DWRule struct {
|
||||||
|
rule byte
|
||||||
|
offset int64
|
||||||
|
newreg uint64
|
||||||
|
expression []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
type FrameContext struct {
|
||||||
|
loc uint64
|
||||||
|
order binary.ByteOrder
|
||||||
|
address uint64
|
||||||
|
cfa CurrentFrameAddress
|
||||||
|
regs map[uint64]DWRule
|
||||||
|
initialRegs map[uint64]DWRule
|
||||||
|
prevRegs map[uint64]DWRule
|
||||||
|
buf *bytes.Buffer
|
||||||
|
cie *CommonInformationEntry
|
||||||
|
codeAlignment uint64
|
||||||
|
dataAlignment int64
|
||||||
|
}
|
||||||
|
|
||||||
|
func (fctx *FrameContext) CFAOffset() int64 {
|
||||||
|
return fctx.cfa.offset
|
||||||
|
}
|
||||||
|
|
||||||
|
// Instructions used to recreate the table from the .debug_frame data.
|
||||||
|
const (
|
||||||
|
DW_CFA_nop = 0x0 // No ops
|
||||||
|
DW_CFA_set_loc = 0x01 // op1: address
|
||||||
|
DW_CFA_advance_loc1 = iota // op1: 1-bytes delta
|
||||||
|
DW_CFA_advance_loc2 // op1: 2-byte delta
|
||||||
|
DW_CFA_advance_loc4 // op1: 4-byte delta
|
||||||
|
DW_CFA_offset_extended // op1: ULEB128 register, op2: ULEB128 offset
|
||||||
|
DW_CFA_restore_extended // op1: ULEB128 register
|
||||||
|
DW_CFA_undefined // op1: ULEB128 register
|
||||||
|
DW_CFA_same_value // op1: ULEB128 register
|
||||||
|
DW_CFA_register // op1: ULEB128 register, op2: ULEB128 register
|
||||||
|
DW_CFA_remember_state // No ops
|
||||||
|
DW_CFA_restore_state // No ops
|
||||||
|
DW_CFA_def_cfa // op1: ULEB128 register, op2: ULEB128 offset
|
||||||
|
DW_CFA_def_cfa_register // op1: ULEB128 register
|
||||||
|
DW_CFA_def_cfa_offset // op1: ULEB128 offset
|
||||||
|
DW_CFA_def_cfa_expression // op1: BLOCK
|
||||||
|
DW_CFA_expression // op1: ULEB128 register, op2: BLOCK
|
||||||
|
DW_CFA_offset_extended_sf // op1: ULEB128 register, op2: SLEB128 BLOCK
|
||||||
|
DW_CFA_def_cfa_sf // op1: ULEB128 register, op2: SLEB128 offset
|
||||||
|
DW_CFA_def_cfa_offset_sf // op1: SLEB128 offset
|
||||||
|
DW_CFA_val_offset // op1: ULEB128, op2: ULEB128
|
||||||
|
DW_CFA_val_offset_sf // op1: ULEB128, op2: SLEB128
|
||||||
|
DW_CFA_val_expression // op1: ULEB128, op2: BLOCK
|
||||||
|
DW_CFA_lo_user = 0x1c // op1: BLOCK
|
||||||
|
DW_CFA_hi_user = 0x3f // op1: ULEB128 register, op2: BLOCK
|
||||||
|
DW_CFA_advance_loc = (0x1 << 6) // High 2 bits: 0x1, low 6: delta
|
||||||
|
DW_CFA_offset = (0x2 << 6) // High 2 bits: 0x2, low 6: register
|
||||||
|
DW_CFA_restore = (0x3 << 6) // High 2 bits: 0x3, low 6: register
|
||||||
|
)
|
||||||
|
|
||||||
|
// Rules defined for register values.
|
||||||
|
const (
|
||||||
|
rule_undefined = iota
|
||||||
|
rule_sameval
|
||||||
|
rule_offset
|
||||||
|
rule_valoffset
|
||||||
|
rule_register
|
||||||
|
rule_expression
|
||||||
|
rule_valexpression
|
||||||
|
rule_architectural
|
||||||
|
)
|
||||||
|
|
||||||
|
const low_6_offset = 0x3f
|
||||||
|
|
||||||
|
type instruction func(frame *FrameContext)
|
||||||
|
|
||||||
|
// // Mapping from DWARF opcode to function.
|
||||||
|
var fnlookup = map[byte]instruction{
|
||||||
|
DW_CFA_advance_loc: advanceloc,
|
||||||
|
DW_CFA_offset: offset,
|
||||||
|
DW_CFA_restore: restore,
|
||||||
|
DW_CFA_set_loc: setloc,
|
||||||
|
DW_CFA_advance_loc1: advanceloc1,
|
||||||
|
DW_CFA_advance_loc2: advanceloc2,
|
||||||
|
DW_CFA_advance_loc4: advanceloc4,
|
||||||
|
DW_CFA_offset_extended: offsetextended,
|
||||||
|
DW_CFA_restore_extended: restoreextended,
|
||||||
|
DW_CFA_undefined: undefined,
|
||||||
|
DW_CFA_same_value: samevalue,
|
||||||
|
DW_CFA_register: register,
|
||||||
|
DW_CFA_remember_state: rememberstate,
|
||||||
|
DW_CFA_restore_state: restorestate,
|
||||||
|
DW_CFA_def_cfa: defcfa,
|
||||||
|
DW_CFA_def_cfa_register: defcfaregister,
|
||||||
|
DW_CFA_def_cfa_offset: defcfaoffset,
|
||||||
|
DW_CFA_def_cfa_expression: defcfaexpression,
|
||||||
|
DW_CFA_expression: expression,
|
||||||
|
DW_CFA_offset_extended_sf: offsetextendedsf,
|
||||||
|
DW_CFA_def_cfa_sf: defcfasf,
|
||||||
|
DW_CFA_def_cfa_offset_sf: defcfaoffsetsf,
|
||||||
|
DW_CFA_val_offset: valoffset,
|
||||||
|
DW_CFA_val_offset_sf: valoffsetsf,
|
||||||
|
DW_CFA_val_expression: valexpression,
|
||||||
|
DW_CFA_lo_user: louser,
|
||||||
|
DW_CFA_hi_user: hiuser,
|
||||||
|
}
|
||||||
|
|
||||||
|
func executeCIEInstructions(cie *CommonInformationEntry) *FrameContext {
|
||||||
|
initialInstructions := make([]byte, len(cie.InitialInstructions))
|
||||||
|
copy(initialInstructions, cie.InitialInstructions)
|
||||||
|
frame := &FrameContext{
|
||||||
|
cie: cie,
|
||||||
|
regs: make(map[uint64]DWRule),
|
||||||
|
initialRegs: make(map[uint64]DWRule),
|
||||||
|
prevRegs: make(map[uint64]DWRule),
|
||||||
|
codeAlignment: cie.CodeAlignmentFactor,
|
||||||
|
dataAlignment: cie.DataAlignmentFactor,
|
||||||
|
buf: bytes.NewBuffer(initialInstructions),
|
||||||
|
}
|
||||||
|
|
||||||
|
frame.ExecuteDwarfProgram()
|
||||||
|
return frame
|
||||||
|
}
|
||||||
|
|
||||||
|
// Unwind the stack to find the return address register.
|
||||||
|
func executeDwarfProgramUntilPC(fde *FrameDescriptionEntry, pc uint64) *FrameContext {
|
||||||
|
frame := executeCIEInstructions(fde.CIE)
|
||||||
|
frame.order = fde.order
|
||||||
|
frame.loc = fde.Begin()
|
||||||
|
frame.address = pc
|
||||||
|
fdeInstructions := make([]byte, len(fde.Instructions))
|
||||||
|
copy(fdeInstructions, fde.Instructions)
|
||||||
|
frame.ExecuteUntilPC(fdeInstructions)
|
||||||
|
|
||||||
|
return frame
|
||||||
|
}
|
||||||
|
|
||||||
|
func (frame *FrameContext) ExecuteDwarfProgram() {
|
||||||
|
for frame.buf.Len() > 0 {
|
||||||
|
executeDwarfInstruction(frame)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Execute dwarf instructions.
|
||||||
|
func (frame *FrameContext) ExecuteUntilPC(instructions []byte) {
|
||||||
|
frame.buf.Truncate(0)
|
||||||
|
frame.buf.Write(instructions)
|
||||||
|
|
||||||
|
// We only need to execute the instructions until
|
||||||
|
// ctx.loc > ctx.addess (which is the address we
|
||||||
|
// are currently at in the traced process).
|
||||||
|
for frame.address >= frame.loc && frame.buf.Len() > 0 {
|
||||||
|
executeDwarfInstruction(frame)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func executeDwarfInstruction(frame *FrameContext) {
|
||||||
|
instruction, err := frame.buf.ReadByte()
|
||||||
|
if err != nil {
|
||||||
|
panic("Could not read from instruction buffer")
|
||||||
|
}
|
||||||
|
|
||||||
|
if instruction == DW_CFA_nop {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
fn := lookupFunc(instruction, frame.buf)
|
||||||
|
|
||||||
|
fn(frame)
|
||||||
|
}
|
||||||
|
|
||||||
|
func lookupFunc(instruction byte, buf *bytes.Buffer) instruction {
|
||||||
|
const high_2_bits = 0xc0
|
||||||
|
var restore bool
|
||||||
|
|
||||||
|
// Special case the 3 opcodes that have their argument encoded in the opcode itself.
|
||||||
|
switch instruction & high_2_bits {
|
||||||
|
case DW_CFA_advance_loc:
|
||||||
|
instruction = DW_CFA_advance_loc
|
||||||
|
restore = true
|
||||||
|
|
||||||
|
case DW_CFA_offset:
|
||||||
|
instruction = DW_CFA_offset
|
||||||
|
restore = true
|
||||||
|
|
||||||
|
case DW_CFA_restore:
|
||||||
|
instruction = DW_CFA_restore
|
||||||
|
restore = true
|
||||||
|
}
|
||||||
|
|
||||||
|
if restore {
|
||||||
|
// Restore the last byte as it actually contains the argument for the opcode.
|
||||||
|
err := buf.UnreadByte()
|
||||||
|
if err != nil {
|
||||||
|
panic("Could not unread byte")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn, ok := fnlookup[instruction]
|
||||||
|
if !ok {
|
||||||
|
panic(fmt.Sprintf("Encountered an unexpected DWARF CFA opcode: %#v", instruction))
|
||||||
|
}
|
||||||
|
|
||||||
|
return fn
|
||||||
|
}
|
||||||
|
|
||||||
|
func advanceloc(frame *FrameContext) {
|
||||||
|
b, err := frame.buf.ReadByte()
|
||||||
|
if err != nil {
|
||||||
|
panic("Could not read byte")
|
||||||
|
}
|
||||||
|
|
||||||
|
delta := b & low_6_offset
|
||||||
|
frame.loc += uint64(delta) * frame.codeAlignment
|
||||||
|
}
|
||||||
|
|
||||||
|
func advanceloc1(frame *FrameContext) {
|
||||||
|
delta, err := frame.buf.ReadByte()
|
||||||
|
if err != nil {
|
||||||
|
panic("Could not read byte")
|
||||||
|
}
|
||||||
|
|
||||||
|
frame.loc += uint64(delta) * frame.codeAlignment
|
||||||
|
}
|
||||||
|
|
||||||
|
func advanceloc2(frame *FrameContext) {
|
||||||
|
var delta uint16
|
||||||
|
binary.Read(frame.buf, frame.order, &delta)
|
||||||
|
|
||||||
|
frame.loc += uint64(delta) * frame.codeAlignment
|
||||||
|
}
|
||||||
|
|
||||||
|
func advanceloc4(frame *FrameContext) {
|
||||||
|
var delta uint32
|
||||||
|
binary.Read(frame.buf, frame.order, &delta)
|
||||||
|
|
||||||
|
frame.loc += uint64(delta) * frame.codeAlignment
|
||||||
|
}
|
||||||
|
|
||||||
|
func offset(frame *FrameContext) {
|
||||||
|
b, err := frame.buf.ReadByte()
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
reg = b & low_6_offset
|
||||||
|
offset, _ = util.DecodeULEB128(frame.buf)
|
||||||
|
)
|
||||||
|
|
||||||
|
frame.regs[uint64(reg)] = DWRule{offset: int64(offset) * frame.dataAlignment, rule: rule_offset}
|
||||||
|
}
|
||||||
|
|
||||||
|
func restore(frame *FrameContext) {
|
||||||
|
b, err := frame.buf.ReadByte()
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
reg := uint64(b & low_6_offset)
|
||||||
|
oldrule, ok := frame.initialRegs[reg]
|
||||||
|
if ok {
|
||||||
|
frame.regs[reg] = DWRule{offset: oldrule.offset, rule: rule_offset}
|
||||||
|
} else {
|
||||||
|
frame.regs[reg] = DWRule{rule: rule_undefined}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func setloc(frame *FrameContext) {
|
||||||
|
var loc uint64
|
||||||
|
binary.Read(frame.buf, frame.order, &loc)
|
||||||
|
|
||||||
|
frame.loc = loc
|
||||||
|
}
|
||||||
|
|
||||||
|
func offsetextended(frame *FrameContext) {
|
||||||
|
var (
|
||||||
|
reg, _ = util.DecodeULEB128(frame.buf)
|
||||||
|
offset, _ = util.DecodeULEB128(frame.buf)
|
||||||
|
)
|
||||||
|
|
||||||
|
frame.regs[reg] = DWRule{offset: int64(offset) * frame.dataAlignment, rule: rule_offset}
|
||||||
|
}
|
||||||
|
|
||||||
|
func undefined(frame *FrameContext) {
|
||||||
|
reg, _ := util.DecodeULEB128(frame.buf)
|
||||||
|
frame.regs[reg] = DWRule{rule: rule_undefined}
|
||||||
|
}
|
||||||
|
|
||||||
|
func samevalue(frame *FrameContext) {
|
||||||
|
reg, _ := util.DecodeULEB128(frame.buf)
|
||||||
|
frame.regs[reg] = DWRule{rule: rule_sameval}
|
||||||
|
}
|
||||||
|
|
||||||
|
func register(frame *FrameContext) {
|
||||||
|
reg1, _ := util.DecodeULEB128(frame.buf)
|
||||||
|
reg2, _ := util.DecodeULEB128(frame.buf)
|
||||||
|
frame.regs[reg1] = DWRule{newreg: reg2, rule: rule_register}
|
||||||
|
}
|
||||||
|
|
||||||
|
func rememberstate(frame *FrameContext) {
|
||||||
|
frame.prevRegs = frame.regs
|
||||||
|
}
|
||||||
|
|
||||||
|
func restorestate(frame *FrameContext) {
|
||||||
|
frame.regs = frame.prevRegs
|
||||||
|
}
|
||||||
|
|
||||||
|
func restoreextended(frame *FrameContext) {
|
||||||
|
reg, _ := util.DecodeULEB128(frame.buf)
|
||||||
|
|
||||||
|
oldrule, ok := frame.initialRegs[reg]
|
||||||
|
if ok {
|
||||||
|
frame.regs[reg] = DWRule{offset: oldrule.offset, rule: rule_offset}
|
||||||
|
} else {
|
||||||
|
frame.regs[reg] = DWRule{rule: rule_undefined}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func defcfa(frame *FrameContext) {
|
||||||
|
reg, _ := util.DecodeULEB128(frame.buf)
|
||||||
|
offset, _ := util.DecodeULEB128(frame.buf)
|
||||||
|
|
||||||
|
frame.cfa.register = reg
|
||||||
|
frame.cfa.offset = int64(offset)
|
||||||
|
}
|
||||||
|
|
||||||
|
func defcfaregister(frame *FrameContext) {
|
||||||
|
reg, _ := util.DecodeULEB128(frame.buf)
|
||||||
|
frame.cfa.register = reg
|
||||||
|
}
|
||||||
|
|
||||||
|
func defcfaoffset(frame *FrameContext) {
|
||||||
|
offset, _ := util.DecodeULEB128(frame.buf)
|
||||||
|
frame.cfa.offset = int64(offset)
|
||||||
|
}
|
||||||
|
|
||||||
|
func defcfasf(frame *FrameContext) {
|
||||||
|
reg, _ := util.DecodeULEB128(frame.buf)
|
||||||
|
offset, _ := util.DecodeSLEB128(frame.buf)
|
||||||
|
|
||||||
|
frame.cfa.register = reg
|
||||||
|
frame.cfa.offset = offset * frame.dataAlignment
|
||||||
|
}
|
||||||
|
|
||||||
|
func defcfaoffsetsf(frame *FrameContext) {
|
||||||
|
offset, _ := util.DecodeSLEB128(frame.buf)
|
||||||
|
offset *= frame.dataAlignment
|
||||||
|
frame.cfa.offset = offset
|
||||||
|
}
|
||||||
|
|
||||||
|
func defcfaexpression(frame *FrameContext) {
|
||||||
|
var (
|
||||||
|
l, _ = util.DecodeULEB128(frame.buf)
|
||||||
|
expr = frame.buf.Next(int(l))
|
||||||
|
)
|
||||||
|
|
||||||
|
frame.cfa.expression = expr
|
||||||
|
frame.cfa.rule = rule_expression
|
||||||
|
}
|
||||||
|
|
||||||
|
func expression(frame *FrameContext) {
|
||||||
|
var (
|
||||||
|
reg, _ = util.DecodeULEB128(frame.buf)
|
||||||
|
l, _ = util.DecodeULEB128(frame.buf)
|
||||||
|
expr = frame.buf.Next(int(l))
|
||||||
|
)
|
||||||
|
|
||||||
|
frame.regs[reg] = DWRule{rule: rule_expression, expression: expr}
|
||||||
|
}
|
||||||
|
|
||||||
|
func offsetextendedsf(frame *FrameContext) {
|
||||||
|
var (
|
||||||
|
reg, _ = util.DecodeULEB128(frame.buf)
|
||||||
|
offset, _ = util.DecodeSLEB128(frame.buf)
|
||||||
|
)
|
||||||
|
|
||||||
|
frame.regs[reg] = DWRule{offset: offset * frame.dataAlignment, rule: rule_offset}
|
||||||
|
}
|
||||||
|
|
||||||
|
func valoffset(frame *FrameContext) {
|
||||||
|
var (
|
||||||
|
reg, _ = util.DecodeULEB128(frame.buf)
|
||||||
|
offset, _ = util.DecodeULEB128(frame.buf)
|
||||||
|
)
|
||||||
|
|
||||||
|
frame.regs[reg] = DWRule{offset: int64(offset), rule: rule_valoffset}
|
||||||
|
}
|
||||||
|
|
||||||
|
func valoffsetsf(frame *FrameContext) {
|
||||||
|
var (
|
||||||
|
reg, _ = util.DecodeULEB128(frame.buf)
|
||||||
|
offset, _ = util.DecodeSLEB128(frame.buf)
|
||||||
|
)
|
||||||
|
|
||||||
|
frame.regs[reg] = DWRule{offset: offset * frame.dataAlignment, rule: rule_valoffset}
|
||||||
|
}
|
||||||
|
|
||||||
|
func valexpression(frame *FrameContext) {
|
||||||
|
var (
|
||||||
|
reg, _ = util.DecodeULEB128(frame.buf)
|
||||||
|
l, _ = util.DecodeULEB128(frame.buf)
|
||||||
|
expr = frame.buf.Next(int(l))
|
||||||
|
)
|
||||||
|
|
||||||
|
frame.regs[reg] = DWRule{rule: rule_valexpression, expression: expr}
|
||||||
|
}
|
||||||
|
|
||||||
|
func louser(frame *FrameContext) {
|
||||||
|
frame.buf.Next(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func hiuser(frame *FrameContext) {
|
||||||
|
frame.buf.Next(1)
|
||||||
|
}
|
122
vendor/github.com/derekparker/delve/pkg/dwarf/line/line_parser.go
generated
vendored
Normal file
122
vendor/github.com/derekparker/delve/pkg/dwarf/line/line_parser.go
generated
vendored
Normal file
@ -0,0 +1,122 @@
|
|||||||
|
package line
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/binary"
|
||||||
|
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/util"
|
||||||
|
)
|
||||||
|
|
||||||
|
type DebugLinePrologue struct {
|
||||||
|
UnitLength uint32
|
||||||
|
Version uint16
|
||||||
|
Length uint32
|
||||||
|
MinInstrLength uint8
|
||||||
|
InitialIsStmt uint8
|
||||||
|
LineBase int8
|
||||||
|
LineRange uint8
|
||||||
|
OpcodeBase uint8
|
||||||
|
StdOpLengths []uint8
|
||||||
|
}
|
||||||
|
|
||||||
|
type DebugLineInfo struct {
|
||||||
|
Prologue *DebugLinePrologue
|
||||||
|
IncludeDirs []string
|
||||||
|
FileNames []*FileEntry
|
||||||
|
Instructions []byte
|
||||||
|
Lookup map[string]*FileEntry
|
||||||
|
}
|
||||||
|
|
||||||
|
type FileEntry struct {
|
||||||
|
Name string
|
||||||
|
DirIdx uint64
|
||||||
|
LastModTime uint64
|
||||||
|
Length uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
type DebugLines []*DebugLineInfo
|
||||||
|
|
||||||
|
func (d *DebugLines) GetLineInfo(name string) *DebugLineInfo {
|
||||||
|
// Find in which table file exists and return it.
|
||||||
|
for _, l := range *d {
|
||||||
|
if _, ok := l.Lookup[name]; ok {
|
||||||
|
return l
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func Parse(data []byte) DebugLines {
|
||||||
|
var (
|
||||||
|
lines = make(DebugLines, 0)
|
||||||
|
buf = bytes.NewBuffer(data)
|
||||||
|
)
|
||||||
|
|
||||||
|
// We have to parse multiple file name tables here.
|
||||||
|
for buf.Len() > 0 {
|
||||||
|
dbl := new(DebugLineInfo)
|
||||||
|
dbl.Lookup = make(map[string]*FileEntry)
|
||||||
|
|
||||||
|
parseDebugLinePrologue(dbl, buf)
|
||||||
|
parseIncludeDirs(dbl, buf)
|
||||||
|
parseFileEntries(dbl, buf)
|
||||||
|
|
||||||
|
// Instructions size calculation breakdown:
|
||||||
|
// - dbl.Prologue.UnitLength is the length of the entire unit, not including the 4 bytes to represent that length.
|
||||||
|
// - dbl.Prologue.Length is the length of the prologue not including unit length, version or prologue length itself.
|
||||||
|
// - So you have UnitLength - PrologueLength - (version_length_bytes(2) + prologue_length_bytes(4)).
|
||||||
|
dbl.Instructions = buf.Next(int(dbl.Prologue.UnitLength - dbl.Prologue.Length - 6))
|
||||||
|
|
||||||
|
lines = append(lines, dbl)
|
||||||
|
}
|
||||||
|
|
||||||
|
return lines
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseDebugLinePrologue(dbl *DebugLineInfo, buf *bytes.Buffer) {
|
||||||
|
p := new(DebugLinePrologue)
|
||||||
|
|
||||||
|
p.UnitLength = binary.LittleEndian.Uint32(buf.Next(4))
|
||||||
|
p.Version = binary.LittleEndian.Uint16(buf.Next(2))
|
||||||
|
p.Length = binary.LittleEndian.Uint32(buf.Next(4))
|
||||||
|
p.MinInstrLength = uint8(buf.Next(1)[0])
|
||||||
|
p.InitialIsStmt = uint8(buf.Next(1)[0])
|
||||||
|
p.LineBase = int8(buf.Next(1)[0])
|
||||||
|
p.LineRange = uint8(buf.Next(1)[0])
|
||||||
|
p.OpcodeBase = uint8(buf.Next(1)[0])
|
||||||
|
|
||||||
|
p.StdOpLengths = make([]uint8, p.OpcodeBase-1)
|
||||||
|
binary.Read(buf, binary.LittleEndian, &p.StdOpLengths)
|
||||||
|
|
||||||
|
dbl.Prologue = p
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseIncludeDirs(info *DebugLineInfo, buf *bytes.Buffer) {
|
||||||
|
for {
|
||||||
|
str, _ := util.ParseString(buf)
|
||||||
|
if str == "" {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
info.IncludeDirs = append(info.IncludeDirs, str)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseFileEntries(info *DebugLineInfo, buf *bytes.Buffer) {
|
||||||
|
for {
|
||||||
|
entry := new(FileEntry)
|
||||||
|
|
||||||
|
name, _ := util.ParseString(buf)
|
||||||
|
if name == "" {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
entry.Name = name
|
||||||
|
entry.DirIdx, _ = util.DecodeULEB128(buf)
|
||||||
|
entry.LastModTime, _ = util.DecodeULEB128(buf)
|
||||||
|
entry.Length, _ = util.DecodeULEB128(buf)
|
||||||
|
|
||||||
|
info.FileNames = append(info.FileNames, entry)
|
||||||
|
info.Lookup[name] = entry
|
||||||
|
}
|
||||||
|
}
|
267
vendor/github.com/derekparker/delve/pkg/dwarf/line/state_machine.go
generated
vendored
Normal file
267
vendor/github.com/derekparker/delve/pkg/dwarf/line/state_machine.go
generated
vendored
Normal file
@ -0,0 +1,267 @@
|
|||||||
|
package line
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/binary"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/util"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Location struct {
|
||||||
|
File string
|
||||||
|
Line int
|
||||||
|
Address uint64
|
||||||
|
Delta int
|
||||||
|
}
|
||||||
|
|
||||||
|
type StateMachine struct {
|
||||||
|
dbl *DebugLineInfo
|
||||||
|
file string
|
||||||
|
line int
|
||||||
|
address uint64
|
||||||
|
column uint
|
||||||
|
isStmt bool
|
||||||
|
basicBlock bool
|
||||||
|
endSeq bool
|
||||||
|
lastWasStandard bool
|
||||||
|
lastDelta int
|
||||||
|
// valid is true if the current value of the state machine is the address of
|
||||||
|
// an instruction (using the terminology used by DWARF spec the current
|
||||||
|
// value of the state machine should be appended to the matrix representing
|
||||||
|
// the compilation unit)
|
||||||
|
valid bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type opcodefn func(*StateMachine, *bytes.Buffer)
|
||||||
|
|
||||||
|
// Special opcodes
|
||||||
|
const (
|
||||||
|
DW_LNS_copy = 1
|
||||||
|
DW_LNS_advance_pc = 2
|
||||||
|
DW_LNS_advance_line = 3
|
||||||
|
DW_LNS_set_file = 4
|
||||||
|
DW_LNS_set_column = 5
|
||||||
|
DW_LNS_negate_stmt = 6
|
||||||
|
DW_LNS_set_basic_block = 7
|
||||||
|
DW_LNS_const_add_pc = 8
|
||||||
|
DW_LNS_fixed_advance_pc = 9
|
||||||
|
)
|
||||||
|
|
||||||
|
// Extended opcodes
|
||||||
|
const (
|
||||||
|
DW_LINE_end_sequence = 1
|
||||||
|
DW_LINE_set_address = 2
|
||||||
|
DW_LINE_define_file = 3
|
||||||
|
)
|
||||||
|
|
||||||
|
var standardopcodes = map[byte]opcodefn{
|
||||||
|
DW_LNS_copy: copyfn,
|
||||||
|
DW_LNS_advance_pc: advancepc,
|
||||||
|
DW_LNS_advance_line: advanceline,
|
||||||
|
DW_LNS_set_file: setfile,
|
||||||
|
DW_LNS_set_column: setcolumn,
|
||||||
|
DW_LNS_negate_stmt: negatestmt,
|
||||||
|
DW_LNS_set_basic_block: setbasicblock,
|
||||||
|
DW_LNS_const_add_pc: constaddpc,
|
||||||
|
DW_LNS_fixed_advance_pc: fixedadvancepc,
|
||||||
|
}
|
||||||
|
|
||||||
|
var extendedopcodes = map[byte]opcodefn{
|
||||||
|
DW_LINE_end_sequence: endsequence,
|
||||||
|
DW_LINE_set_address: setaddress,
|
||||||
|
DW_LINE_define_file: definefile,
|
||||||
|
}
|
||||||
|
|
||||||
|
func newStateMachine(dbl *DebugLineInfo) *StateMachine {
|
||||||
|
return &StateMachine{dbl: dbl, file: dbl.FileNames[0].Name, line: 1}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns all PCs for a given file/line. Useful for loops where the 'for' line
|
||||||
|
// could be split amongst 2 PCs.
|
||||||
|
func (dbl *DebugLines) AllPCsForFileLine(f string, l int) (pcs []uint64) {
|
||||||
|
var (
|
||||||
|
foundFile bool
|
||||||
|
lastAddr uint64
|
||||||
|
lineInfo = dbl.GetLineInfo(f)
|
||||||
|
sm = newStateMachine(lineInfo)
|
||||||
|
buf = bytes.NewBuffer(lineInfo.Instructions)
|
||||||
|
)
|
||||||
|
|
||||||
|
for b, err := buf.ReadByte(); err == nil; b, err = buf.ReadByte() {
|
||||||
|
findAndExecOpcode(sm, buf, b)
|
||||||
|
if foundFile && sm.file != f {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if sm.line == l && sm.file == f && sm.address != lastAddr {
|
||||||
|
foundFile = true
|
||||||
|
if sm.valid {
|
||||||
|
pcs = append(pcs, sm.address)
|
||||||
|
}
|
||||||
|
line := sm.line
|
||||||
|
// Keep going until we're on a different line. We only care about
|
||||||
|
// when a line comes back around (i.e. for loop) so get to next line,
|
||||||
|
// and try to find the line we care about again.
|
||||||
|
for b, err := buf.ReadByte(); err == nil; b, err = buf.ReadByte() {
|
||||||
|
findAndExecOpcode(sm, buf, b)
|
||||||
|
if line < sm.line {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var NoSourceError = errors.New("no source available")
|
||||||
|
|
||||||
|
func (dbl *DebugLines) AllPCsBetween(begin, end uint64, filename string) ([]uint64, error) {
|
||||||
|
lineInfo := dbl.GetLineInfo(filename)
|
||||||
|
if lineInfo == nil {
|
||||||
|
return nil, NoSourceError
|
||||||
|
}
|
||||||
|
var (
|
||||||
|
pcs []uint64
|
||||||
|
lastaddr uint64
|
||||||
|
sm = newStateMachine(lineInfo)
|
||||||
|
buf = bytes.NewBuffer(lineInfo.Instructions)
|
||||||
|
)
|
||||||
|
|
||||||
|
for b, err := buf.ReadByte(); err == nil; b, err = buf.ReadByte() {
|
||||||
|
findAndExecOpcode(sm, buf, b)
|
||||||
|
if !sm.valid {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if sm.address > end {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if sm.address >= begin && sm.address > lastaddr {
|
||||||
|
lastaddr = sm.address
|
||||||
|
pcs = append(pcs, sm.address)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return pcs, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func findAndExecOpcode(sm *StateMachine, buf *bytes.Buffer, b byte) {
|
||||||
|
switch {
|
||||||
|
case b == 0:
|
||||||
|
execExtendedOpcode(sm, b, buf)
|
||||||
|
case b < sm.dbl.Prologue.OpcodeBase:
|
||||||
|
execStandardOpcode(sm, b, buf)
|
||||||
|
default:
|
||||||
|
execSpecialOpcode(sm, b)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func execSpecialOpcode(sm *StateMachine, instr byte) {
|
||||||
|
var (
|
||||||
|
opcode = uint8(instr)
|
||||||
|
decoded = opcode - sm.dbl.Prologue.OpcodeBase
|
||||||
|
)
|
||||||
|
|
||||||
|
if sm.dbl.Prologue.InitialIsStmt == uint8(1) {
|
||||||
|
sm.isStmt = true
|
||||||
|
}
|
||||||
|
|
||||||
|
sm.lastDelta = int(sm.dbl.Prologue.LineBase + int8(decoded%sm.dbl.Prologue.LineRange))
|
||||||
|
sm.line += sm.lastDelta
|
||||||
|
sm.address += uint64(decoded/sm.dbl.Prologue.LineRange) * uint64(sm.dbl.Prologue.MinInstrLength)
|
||||||
|
sm.basicBlock = false
|
||||||
|
sm.lastWasStandard = false
|
||||||
|
sm.valid = true
|
||||||
|
}
|
||||||
|
|
||||||
|
func execExtendedOpcode(sm *StateMachine, instr byte, buf *bytes.Buffer) {
|
||||||
|
_, _ = util.DecodeULEB128(buf)
|
||||||
|
b, _ := buf.ReadByte()
|
||||||
|
fn, ok := extendedopcodes[b]
|
||||||
|
if !ok {
|
||||||
|
panic(fmt.Sprintf("Encountered unknown extended opcode %#v\n", b))
|
||||||
|
}
|
||||||
|
sm.lastWasStandard = false
|
||||||
|
sm.valid = false
|
||||||
|
|
||||||
|
fn(sm, buf)
|
||||||
|
}
|
||||||
|
|
||||||
|
func execStandardOpcode(sm *StateMachine, instr byte, buf *bytes.Buffer) {
|
||||||
|
fn, ok := standardopcodes[instr]
|
||||||
|
if !ok {
|
||||||
|
panic(fmt.Sprintf("Encountered unknown standard opcode %#v\n", instr))
|
||||||
|
}
|
||||||
|
sm.lastWasStandard = true
|
||||||
|
sm.valid = false
|
||||||
|
|
||||||
|
fn(sm, buf)
|
||||||
|
}
|
||||||
|
|
||||||
|
func copyfn(sm *StateMachine, buf *bytes.Buffer) {
|
||||||
|
sm.basicBlock = false
|
||||||
|
sm.valid = true
|
||||||
|
}
|
||||||
|
|
||||||
|
func advancepc(sm *StateMachine, buf *bytes.Buffer) {
|
||||||
|
addr, _ := util.DecodeULEB128(buf)
|
||||||
|
sm.address += addr * uint64(sm.dbl.Prologue.MinInstrLength)
|
||||||
|
}
|
||||||
|
|
||||||
|
func advanceline(sm *StateMachine, buf *bytes.Buffer) {
|
||||||
|
line, _ := util.DecodeSLEB128(buf)
|
||||||
|
sm.line += int(line)
|
||||||
|
sm.lastDelta = int(line)
|
||||||
|
}
|
||||||
|
|
||||||
|
func setfile(sm *StateMachine, buf *bytes.Buffer) {
|
||||||
|
i, _ := util.DecodeULEB128(buf)
|
||||||
|
sm.file = sm.dbl.FileNames[i-1].Name
|
||||||
|
}
|
||||||
|
|
||||||
|
func setcolumn(sm *StateMachine, buf *bytes.Buffer) {
|
||||||
|
c, _ := util.DecodeULEB128(buf)
|
||||||
|
sm.column = uint(c)
|
||||||
|
}
|
||||||
|
|
||||||
|
func negatestmt(sm *StateMachine, buf *bytes.Buffer) {
|
||||||
|
sm.isStmt = !sm.isStmt
|
||||||
|
}
|
||||||
|
|
||||||
|
func setbasicblock(sm *StateMachine, buf *bytes.Buffer) {
|
||||||
|
sm.basicBlock = true
|
||||||
|
}
|
||||||
|
|
||||||
|
func constaddpc(sm *StateMachine, buf *bytes.Buffer) {
|
||||||
|
sm.address += uint64((255-sm.dbl.Prologue.OpcodeBase)/sm.dbl.Prologue.LineRange) * uint64(sm.dbl.Prologue.MinInstrLength)
|
||||||
|
}
|
||||||
|
|
||||||
|
func fixedadvancepc(sm *StateMachine, buf *bytes.Buffer) {
|
||||||
|
var operand uint16
|
||||||
|
binary.Read(buf, binary.LittleEndian, &operand)
|
||||||
|
|
||||||
|
sm.address += uint64(operand)
|
||||||
|
}
|
||||||
|
|
||||||
|
func endsequence(sm *StateMachine, buf *bytes.Buffer) {
|
||||||
|
sm.endSeq = true
|
||||||
|
sm.valid = true
|
||||||
|
}
|
||||||
|
|
||||||
|
func setaddress(sm *StateMachine, buf *bytes.Buffer) {
|
||||||
|
var addr uint64
|
||||||
|
|
||||||
|
binary.Read(buf, binary.LittleEndian, &addr)
|
||||||
|
|
||||||
|
sm.address = addr
|
||||||
|
}
|
||||||
|
|
||||||
|
func definefile(sm *StateMachine, buf *bytes.Buffer) {
|
||||||
|
var (
|
||||||
|
_, _ = util.ParseString(buf)
|
||||||
|
_, _ = util.DecodeULEB128(buf)
|
||||||
|
_, _ = util.DecodeULEB128(buf)
|
||||||
|
_, _ = util.DecodeULEB128(buf)
|
||||||
|
)
|
||||||
|
|
||||||
|
// Don't do anything here yet.
|
||||||
|
}
|
84
vendor/github.com/derekparker/delve/pkg/dwarf/op/op.go
generated
vendored
Normal file
84
vendor/github.com/derekparker/delve/pkg/dwarf/op/op.go
generated
vendored
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
package op
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/binary"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/util"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
DW_OP_addr = 0x3
|
||||||
|
DW_OP_call_frame_cfa = 0x9c
|
||||||
|
DW_OP_plus = 0x22
|
||||||
|
DW_OP_consts = 0x11
|
||||||
|
DW_OP_plus_uconsts = 0x23
|
||||||
|
)
|
||||||
|
|
||||||
|
type stackfn func(*bytes.Buffer, []int64, int64) ([]int64, error)
|
||||||
|
|
||||||
|
var oplut = map[byte]stackfn{
|
||||||
|
DW_OP_call_frame_cfa: callframecfa,
|
||||||
|
DW_OP_plus: plus,
|
||||||
|
DW_OP_consts: consts,
|
||||||
|
DW_OP_addr: addr,
|
||||||
|
DW_OP_plus_uconsts: plusuconsts,
|
||||||
|
}
|
||||||
|
|
||||||
|
func ExecuteStackProgram(cfa int64, instructions []byte) (int64, error) {
|
||||||
|
stack := make([]int64, 0, 3)
|
||||||
|
buf := bytes.NewBuffer(instructions)
|
||||||
|
|
||||||
|
for opcode, err := buf.ReadByte(); err == nil; opcode, err = buf.ReadByte() {
|
||||||
|
fn, ok := oplut[opcode]
|
||||||
|
if !ok {
|
||||||
|
return 0, fmt.Errorf("invalid instruction %#v", opcode)
|
||||||
|
}
|
||||||
|
|
||||||
|
stack, err = fn(buf, stack, cfa)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(stack) == 0 {
|
||||||
|
return 0, errors.New("empty OP stack")
|
||||||
|
}
|
||||||
|
|
||||||
|
return stack[len(stack)-1], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func callframecfa(buf *bytes.Buffer, stack []int64, cfa int64) ([]int64, error) {
|
||||||
|
if cfa == 0 {
|
||||||
|
return stack, fmt.Errorf("Could not retrieve CFA for current PC")
|
||||||
|
}
|
||||||
|
return append(stack, int64(cfa)), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func addr(buf *bytes.Buffer, stack []int64, cfa int64) ([]int64, error) {
|
||||||
|
return append(stack, int64(binary.LittleEndian.Uint64(buf.Next(8)))), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func plus(buf *bytes.Buffer, stack []int64, cfa int64) ([]int64, error) {
|
||||||
|
var (
|
||||||
|
slen = len(stack)
|
||||||
|
digits = stack[slen-2 : slen]
|
||||||
|
st = stack[:slen-2]
|
||||||
|
)
|
||||||
|
|
||||||
|
return append(st, digits[0]+digits[1]), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func plusuconsts(buf *bytes.Buffer, stack []int64, cfa int64) ([]int64, error) {
|
||||||
|
slen := len(stack)
|
||||||
|
num, _ := util.DecodeULEB128(buf)
|
||||||
|
stack[slen-1] = stack[slen-1] + int64(num)
|
||||||
|
return stack, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func consts(buf *bytes.Buffer, stack []int64, cfa int64) ([]int64, error) {
|
||||||
|
num, _ := util.DecodeSLEB128(buf)
|
||||||
|
return append(stack, num), nil
|
||||||
|
}
|
345
vendor/github.com/derekparker/delve/pkg/dwarf/reader/reader.go
generated
vendored
Executable file
345
vendor/github.com/derekparker/delve/pkg/dwarf/reader/reader.go
generated
vendored
Executable file
@ -0,0 +1,345 @@
|
|||||||
|
package reader
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/op"
|
||||||
|
"golang.org/x/debug/dwarf"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Reader struct {
|
||||||
|
*dwarf.Reader
|
||||||
|
depth int
|
||||||
|
}
|
||||||
|
|
||||||
|
// New returns a reader for the specified dwarf data.
|
||||||
|
func New(data *dwarf.Data) *Reader {
|
||||||
|
return &Reader{data.Reader(), 0}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Seek moves the reader to an arbitrary offset.
|
||||||
|
func (reader *Reader) Seek(off dwarf.Offset) {
|
||||||
|
reader.depth = 0
|
||||||
|
reader.Reader.Seek(off)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SeekToEntry moves the reader to an arbitrary entry.
|
||||||
|
func (reader *Reader) SeekToEntry(entry *dwarf.Entry) error {
|
||||||
|
reader.Seek(entry.Offset)
|
||||||
|
// Consume the current entry so .Next works as intended
|
||||||
|
_, err := reader.Next()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// SeekToFunctionEntry moves the reader to the function that includes the
|
||||||
|
// specified program counter.
|
||||||
|
func (reader *Reader) SeekToFunction(pc uint64) (*dwarf.Entry, error) {
|
||||||
|
reader.Seek(0)
|
||||||
|
for entry, err := reader.Next(); entry != nil; entry, err = reader.Next() {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if entry.Tag != dwarf.TagSubprogram {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
lowpc, ok := entry.Val(dwarf.AttrLowpc).(uint64)
|
||||||
|
if !ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
highpc, ok := entry.Val(dwarf.AttrHighpc).(uint64)
|
||||||
|
if !ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if lowpc <= pc && highpc > pc {
|
||||||
|
return entry, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, fmt.Errorf("unable to find function context")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns the address for the named entry.
|
||||||
|
func (reader *Reader) AddrFor(name string) (uint64, error) {
|
||||||
|
entry, err := reader.FindEntryNamed(name, false)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
instructions, ok := entry.Val(dwarf.AttrLocation).([]byte)
|
||||||
|
if !ok {
|
||||||
|
return 0, fmt.Errorf("type assertion failed")
|
||||||
|
}
|
||||||
|
addr, err := op.ExecuteStackProgram(0, instructions)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
return uint64(addr), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns the address for the named struct member. Expects the reader to be at the parent entry
|
||||||
|
// or one of the parents children, thus does not seek to parent by itself.
|
||||||
|
func (reader *Reader) AddrForMember(member string, initialInstructions []byte) (uint64, error) {
|
||||||
|
for {
|
||||||
|
entry, err := reader.NextMemberVariable()
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
if entry == nil {
|
||||||
|
return 0, fmt.Errorf("nil entry for member named %s", member)
|
||||||
|
}
|
||||||
|
name, ok := entry.Val(dwarf.AttrName).(string)
|
||||||
|
if !ok || name != member {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
instructions, ok := entry.Val(dwarf.AttrDataMemberLoc).([]byte)
|
||||||
|
if !ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
addr, err := op.ExecuteStackProgram(0, append(initialInstructions, instructions...))
|
||||||
|
return uint64(addr), err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var TypeNotFoundErr = errors.New("no type entry found, use 'types' for a list of valid types")
|
||||||
|
|
||||||
|
// SeekToType moves the reader to the type specified by the entry,
|
||||||
|
// optionally resolving typedefs and pointer types. If the reader is set
|
||||||
|
// to a struct type the NextMemberVariable call can be used to walk all member data.
|
||||||
|
func (reader *Reader) SeekToType(entry *dwarf.Entry, resolveTypedefs bool, resolvePointerTypes bool) (*dwarf.Entry, error) {
|
||||||
|
offset, ok := entry.Val(dwarf.AttrType).(dwarf.Offset)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("entry does not have a type attribute")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Seek to the first type offset
|
||||||
|
reader.Seek(offset)
|
||||||
|
|
||||||
|
// Walk the types to the base
|
||||||
|
for typeEntry, err := reader.Next(); typeEntry != nil; typeEntry, err = reader.Next() {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if typeEntry.Tag == dwarf.TagTypedef && !resolveTypedefs {
|
||||||
|
return typeEntry, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if typeEntry.Tag == dwarf.TagPointerType && !resolvePointerTypes {
|
||||||
|
return typeEntry, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
offset, ok = typeEntry.Val(dwarf.AttrType).(dwarf.Offset)
|
||||||
|
if !ok {
|
||||||
|
return typeEntry, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
reader.Seek(offset)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, TypeNotFoundErr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (reader *Reader) NextType() (*dwarf.Entry, error) {
|
||||||
|
for entry, err := reader.Next(); entry != nil; entry, err = reader.Next() {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
switch entry.Tag {
|
||||||
|
case dwarf.TagArrayType, dwarf.TagBaseType, dwarf.TagClassType, dwarf.TagStructType, dwarf.TagUnionType, dwarf.TagConstType, dwarf.TagVolatileType, dwarf.TagRestrictType, dwarf.TagEnumerationType, dwarf.TagPointerType, dwarf.TagSubroutineType, dwarf.TagTypedef, dwarf.TagUnspecifiedType:
|
||||||
|
return entry, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SeekToTypeNamed moves the reader to the type specified by the name.
|
||||||
|
// If the reader is set to a struct type the NextMemberVariable call
|
||||||
|
// can be used to walk all member data.
|
||||||
|
func (reader *Reader) SeekToTypeNamed(name string) (*dwarf.Entry, error) {
|
||||||
|
// Walk the types to the base
|
||||||
|
for entry, err := reader.NextType(); entry != nil; entry, err = reader.NextType() {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
n, ok := entry.Val(dwarf.AttrName).(string)
|
||||||
|
if !ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if n == name {
|
||||||
|
return entry, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, TypeNotFoundErr
|
||||||
|
}
|
||||||
|
|
||||||
|
// Finds the entry for 'name'.
|
||||||
|
func (reader *Reader) FindEntryNamed(name string, member bool) (*dwarf.Entry, error) {
|
||||||
|
depth := 1
|
||||||
|
for entry, err := reader.Next(); entry != nil; entry, err = reader.Next() {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if entry.Children {
|
||||||
|
depth++
|
||||||
|
}
|
||||||
|
|
||||||
|
if entry.Tag == 0 {
|
||||||
|
depth--
|
||||||
|
if depth <= 0 {
|
||||||
|
return nil, fmt.Errorf("could not find symbol value for %s", name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if member {
|
||||||
|
if entry.Tag != dwarf.TagMember {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if entry.Tag != dwarf.TagVariable && entry.Tag != dwarf.TagFormalParameter && entry.Tag != dwarf.TagStructType {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
n, ok := entry.Val(dwarf.AttrName).(string)
|
||||||
|
if !ok || n != name {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
return entry, nil
|
||||||
|
}
|
||||||
|
return nil, fmt.Errorf("could not find symbol value for %s", name)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (reader *Reader) InstructionsForEntryNamed(name string, member bool) ([]byte, error) {
|
||||||
|
entry, err := reader.FindEntryNamed(name, member)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var attr dwarf.Attr
|
||||||
|
if member {
|
||||||
|
attr = dwarf.AttrDataMemberLoc
|
||||||
|
} else {
|
||||||
|
attr = dwarf.AttrLocation
|
||||||
|
}
|
||||||
|
instr, ok := entry.Val(attr).([]byte)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("invalid typecast for Dwarf instructions")
|
||||||
|
}
|
||||||
|
return instr, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (reader *Reader) InstructionsForEntry(entry *dwarf.Entry) ([]byte, error) {
|
||||||
|
if entry.Tag == dwarf.TagMember {
|
||||||
|
instructions, ok := entry.Val(dwarf.AttrDataMemberLoc).([]byte)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("member data has no data member location attribute")
|
||||||
|
}
|
||||||
|
// clone slice to prevent stomping on the dwarf data
|
||||||
|
return append([]byte{}, instructions...), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// non-member
|
||||||
|
instructions, ok := entry.Val(dwarf.AttrLocation).([]byte)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("entry has no location attribute")
|
||||||
|
}
|
||||||
|
|
||||||
|
// clone slice to prevent stomping on the dwarf data
|
||||||
|
return append([]byte{}, instructions...), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// NextScopeVariable moves the reader to the next debug entry that describes a local variable and returns the entry.
|
||||||
|
func (reader *Reader) NextScopeVariable() (*dwarf.Entry, error) {
|
||||||
|
for entry, err := reader.Next(); entry != nil; entry, err = reader.Next() {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// All scope variables will be at the same depth
|
||||||
|
reader.SkipChildren()
|
||||||
|
|
||||||
|
// End of the current depth
|
||||||
|
if entry.Tag == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
if entry.Tag == dwarf.TagVariable || entry.Tag == dwarf.TagFormalParameter {
|
||||||
|
return entry, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// No more items
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// NextMememberVariable moves the reader to the next debug entry that describes a member variable and returns the entry.
|
||||||
|
func (reader *Reader) NextMemberVariable() (*dwarf.Entry, error) {
|
||||||
|
for entry, err := reader.Next(); entry != nil; entry, err = reader.Next() {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// All member variables will be at the same depth
|
||||||
|
reader.SkipChildren()
|
||||||
|
|
||||||
|
// End of the current depth
|
||||||
|
if entry.Tag == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
if entry.Tag == dwarf.TagMember {
|
||||||
|
return entry, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// No more items
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// NextPackageVariable moves the reader to the next debug entry that describes a package variable.
|
||||||
|
// Any TagVariable entry that is not inside a sub prgram entry and is marked external is considered a package variable.
|
||||||
|
func (reader *Reader) NextPackageVariable() (*dwarf.Entry, error) {
|
||||||
|
for entry, err := reader.Next(); entry != nil; entry, err = reader.Next() {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if entry.Tag == dwarf.TagVariable {
|
||||||
|
ext, ok := entry.Val(dwarf.AttrExternal).(bool)
|
||||||
|
if ok && ext {
|
||||||
|
return entry, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ignore everything inside sub programs
|
||||||
|
if entry.Tag == dwarf.TagSubprogram {
|
||||||
|
reader.SkipChildren()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// No more items
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (reader *Reader) NextCompileUnit() (*dwarf.Entry, error) {
|
||||||
|
for entry, err := reader.Next(); entry != nil; entry, err = reader.Next() {
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if entry.Tag == dwarf.TagCompileUnit {
|
||||||
|
return entry, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, nil
|
||||||
|
}
|
81
vendor/github.com/derekparker/delve/pkg/dwarf/util/util.go
generated
vendored
Normal file
81
vendor/github.com/derekparker/delve/pkg/dwarf/util/util.go
generated
vendored
Normal file
@ -0,0 +1,81 @@
|
|||||||
|
package util
|
||||||
|
|
||||||
|
import "bytes"
|
||||||
|
|
||||||
|
// DecodeULEB128 decodes an unsigned Little Endian Base 128
|
||||||
|
// represented number.
|
||||||
|
func DecodeULEB128(buf *bytes.Buffer) (uint64, uint32) {
|
||||||
|
var (
|
||||||
|
result uint64
|
||||||
|
shift uint64
|
||||||
|
length uint32
|
||||||
|
)
|
||||||
|
|
||||||
|
if buf.Len() == 0 {
|
||||||
|
return 0, 0
|
||||||
|
}
|
||||||
|
|
||||||
|
for {
|
||||||
|
b, err := buf.ReadByte()
|
||||||
|
if err != nil {
|
||||||
|
panic("Could not parse ULEB128 value")
|
||||||
|
}
|
||||||
|
length++
|
||||||
|
|
||||||
|
result |= uint64((uint(b) & 0x7f) << shift)
|
||||||
|
|
||||||
|
// If high order bit is 1.
|
||||||
|
if b&0x80 == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
shift += 7
|
||||||
|
}
|
||||||
|
|
||||||
|
return result, length
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecodeSLEB128 decodes a signed Little Endian Base 128
|
||||||
|
// represented number.
|
||||||
|
func DecodeSLEB128(buf *bytes.Buffer) (int64, uint32) {
|
||||||
|
var (
|
||||||
|
b byte
|
||||||
|
err error
|
||||||
|
result int64
|
||||||
|
shift uint64
|
||||||
|
length uint32
|
||||||
|
)
|
||||||
|
|
||||||
|
if buf.Len() == 0 {
|
||||||
|
return 0, 0
|
||||||
|
}
|
||||||
|
|
||||||
|
for {
|
||||||
|
b, err = buf.ReadByte()
|
||||||
|
if err != nil {
|
||||||
|
panic("Could not parse SLEB128 value")
|
||||||
|
}
|
||||||
|
length++
|
||||||
|
|
||||||
|
result |= int64((int64(b) & 0x7f) << shift)
|
||||||
|
shift += 7
|
||||||
|
if b&0x80 == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (shift < 8*uint64(length)) && (b&0x40 > 0) {
|
||||||
|
result |= -(1 << shift)
|
||||||
|
}
|
||||||
|
|
||||||
|
return result, length
|
||||||
|
}
|
||||||
|
|
||||||
|
func ParseString(data *bytes.Buffer) (string, uint32) {
|
||||||
|
str, err := data.ReadString(0x0)
|
||||||
|
if err != nil {
|
||||||
|
panic("Could not parse string")
|
||||||
|
}
|
||||||
|
|
||||||
|
return str[:len(str)-1], uint32(len(str))
|
||||||
|
}
|
79
vendor/github.com/derekparker/delve/pkg/proc/arch.go
generated
vendored
Normal file
79
vendor/github.com/derekparker/delve/pkg/proc/arch.go
generated
vendored
Normal file
@ -0,0 +1,79 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import "runtime"
|
||||||
|
|
||||||
|
// Arch defines an interface for representing a
|
||||||
|
// CPU architecture.
|
||||||
|
type Arch interface {
|
||||||
|
SetGStructOffset(ver GoVersion, iscgo bool)
|
||||||
|
PtrSize() int
|
||||||
|
BreakpointInstruction() []byte
|
||||||
|
BreakpointSize() int
|
||||||
|
GStructOffset() uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
// AMD64 represents the AMD64 CPU architecture.
|
||||||
|
type AMD64 struct {
|
||||||
|
ptrSize int
|
||||||
|
breakInstruction []byte
|
||||||
|
breakInstructionLen int
|
||||||
|
gStructOffset uint64
|
||||||
|
hardwareBreakpointUsage []bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// AMD64Arch returns an initialized AMD64
|
||||||
|
// struct.
|
||||||
|
func AMD64Arch() *AMD64 {
|
||||||
|
var breakInstr = []byte{0xCC}
|
||||||
|
|
||||||
|
return &AMD64{
|
||||||
|
ptrSize: 8,
|
||||||
|
breakInstruction: breakInstr,
|
||||||
|
breakInstructionLen: len(breakInstr),
|
||||||
|
hardwareBreakpointUsage: make([]bool, 4),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetGStructOffset sets the offset of the G struct on the AMD64
|
||||||
|
// arch struct. The offset is dependent on the Go compiler Version
|
||||||
|
// and whether or not the target program was externally linked.
|
||||||
|
func (a *AMD64) SetGStructOffset(ver GoVersion, isextld bool) {
|
||||||
|
switch runtime.GOOS {
|
||||||
|
case "darwin":
|
||||||
|
a.gStructOffset = 0x8a0
|
||||||
|
case "linux":
|
||||||
|
a.gStructOffset = 0xfffffffffffffff0
|
||||||
|
if isextld || ver.AfterOrEqual(GoVersion{1, 5, -1, 2, 0}) || ver.IsDevel() {
|
||||||
|
a.gStructOffset += 8
|
||||||
|
}
|
||||||
|
case "windows":
|
||||||
|
// Use ArbitraryUserPointer (0x28) as pointer to pointer
|
||||||
|
// to G struct per:
|
||||||
|
// https://golang.org/src/runtime/cgo/gcc_windows_amd64.c
|
||||||
|
a.gStructOffset = 0x28
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// PtrSize returns the size of a pointer
|
||||||
|
// on this architecture.
|
||||||
|
func (a *AMD64) PtrSize() int {
|
||||||
|
return a.ptrSize
|
||||||
|
}
|
||||||
|
|
||||||
|
// BreakpointInstruction returns the Breakpoint
|
||||||
|
// instruction for this architecture.
|
||||||
|
func (a *AMD64) BreakpointInstruction() []byte {
|
||||||
|
return a.breakInstruction
|
||||||
|
}
|
||||||
|
|
||||||
|
// BreakpointSize returns the size of the
|
||||||
|
// breakpoint instruction on this architecture.
|
||||||
|
func (a *AMD64) BreakpointSize() int {
|
||||||
|
return a.breakInstructionLen
|
||||||
|
}
|
||||||
|
|
||||||
|
// GStructOffset returns the offset of the G
|
||||||
|
// struct in thread local storage.
|
||||||
|
func (a *AMD64) GStructOffset() uint64 {
|
||||||
|
return a.gStructOffset
|
||||||
|
}
|
163
vendor/github.com/derekparker/delve/pkg/proc/breakpoints.go
generated
vendored
Normal file
163
vendor/github.com/derekparker/delve/pkg/proc/breakpoints.go
generated
vendored
Normal file
@ -0,0 +1,163 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"go/ast"
|
||||||
|
"go/constant"
|
||||||
|
"reflect"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Breakpoint represents a breakpoint. Stores information on the break
|
||||||
|
// point including the byte of data that originally was stored at that
|
||||||
|
// address.
|
||||||
|
type Breakpoint struct {
|
||||||
|
// File & line information for printing.
|
||||||
|
FunctionName string
|
||||||
|
File string
|
||||||
|
Line int
|
||||||
|
|
||||||
|
Addr uint64 // Address breakpoint is set for.
|
||||||
|
OriginalData []byte // If software breakpoint, the data we replace with breakpoint instruction.
|
||||||
|
Name string // User defined name of the breakpoint
|
||||||
|
ID int // Monotonically increasing ID.
|
||||||
|
Kind BreakpointKind // Whether this is an internal breakpoint (for next'ing or stepping).
|
||||||
|
|
||||||
|
// Breakpoint information
|
||||||
|
Tracepoint bool // Tracepoint flag
|
||||||
|
Goroutine bool // Retrieve goroutine information
|
||||||
|
Stacktrace int // Number of stack frames to retrieve
|
||||||
|
Variables []string // Variables to evaluate
|
||||||
|
LoadArgs *LoadConfig
|
||||||
|
LoadLocals *LoadConfig
|
||||||
|
HitCount map[int]uint64 // Number of times a breakpoint has been reached in a certain goroutine
|
||||||
|
TotalHitCount uint64 // Number of times a breakpoint has been reached
|
||||||
|
|
||||||
|
// DeferReturns: when kind == NextDeferBreakpoint this breakpoint
|
||||||
|
// will also check if the caller is runtime.gopanic or if the return
|
||||||
|
// address is in the DeferReturns array.
|
||||||
|
// Next uses NextDeferBreakpoints for the breakpoint it sets on the
|
||||||
|
// deferred function, DeferReturns is populated with the
|
||||||
|
// addresses of calls to runtime.deferreturn in the current
|
||||||
|
// function. This insures that the breakpoint on the deferred
|
||||||
|
// function only triggers on panic or on the defer call to
|
||||||
|
// the function, not when the function is called directly
|
||||||
|
DeferReturns []uint64
|
||||||
|
// Cond: if not nil the breakpoint will be triggered only if evaluating Cond returns true
|
||||||
|
Cond ast.Expr
|
||||||
|
}
|
||||||
|
|
||||||
|
// Breakpoint Kind determines the behavior of delve when the
|
||||||
|
// breakpoint is reached.
|
||||||
|
type BreakpointKind int
|
||||||
|
|
||||||
|
const (
|
||||||
|
// UserBreakpoint is a user set breakpoint
|
||||||
|
UserBreakpoint BreakpointKind = iota
|
||||||
|
// NextBreakpoint is a breakpoint set by Next, Continue
|
||||||
|
// will stop on it and delete it
|
||||||
|
NextBreakpoint
|
||||||
|
// NextDeferBreakpoint is a breakpoint set by Next on the
|
||||||
|
// first deferred function. In addition to checking their condition
|
||||||
|
// breakpoints of this kind will also check that the function has been
|
||||||
|
// called by runtime.gopanic or through runtime.deferreturn.
|
||||||
|
NextDeferBreakpoint
|
||||||
|
// StepBreakpoint is a breakpoint set by Step on a CALL instruction,
|
||||||
|
// Continue will set a new breakpoint (of NextBreakpoint kind) on the
|
||||||
|
// destination of CALL, delete this breakpoint and then continue again
|
||||||
|
StepBreakpoint
|
||||||
|
)
|
||||||
|
|
||||||
|
func (bp *Breakpoint) String() string {
|
||||||
|
return fmt.Sprintf("Breakpoint %d at %#v %s:%d (%d)", bp.ID, bp.Addr, bp.File, bp.Line, bp.TotalHitCount)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Clear this breakpoint appropriately depending on whether it is a
|
||||||
|
// hardware or software breakpoint.
|
||||||
|
func (bp *Breakpoint) Clear(thread *Thread) (*Breakpoint, error) {
|
||||||
|
if _, err := thread.writeMemory(uintptr(bp.Addr), bp.OriginalData); err != nil {
|
||||||
|
return nil, fmt.Errorf("could not clear breakpoint %s", err)
|
||||||
|
}
|
||||||
|
return bp, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// BreakpointExistsError is returned when trying to set a breakpoint at
|
||||||
|
// an address that already has a breakpoint set for it.
|
||||||
|
type BreakpointExistsError struct {
|
||||||
|
file string
|
||||||
|
line int
|
||||||
|
addr uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bpe BreakpointExistsError) Error() string {
|
||||||
|
return fmt.Sprintf("Breakpoint exists at %s:%d at %x", bpe.file, bpe.line, bpe.addr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// InvalidAddressError represents the result of
|
||||||
|
// attempting to set a breakpoint at an invalid address.
|
||||||
|
type InvalidAddressError struct {
|
||||||
|
address uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
func (iae InvalidAddressError) Error() string {
|
||||||
|
return fmt.Sprintf("Invalid address %#v\n", iae.address)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) writeSoftwareBreakpoint(thread *Thread, addr uint64) error {
|
||||||
|
_, err := thread.writeMemory(uintptr(addr), dbp.arch.BreakpointInstruction())
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bp *Breakpoint) checkCondition(thread *Thread) (bool, error) {
|
||||||
|
if bp.Cond == nil {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
if bp.Kind == NextDeferBreakpoint {
|
||||||
|
frames, err := thread.Stacktrace(2)
|
||||||
|
if err == nil {
|
||||||
|
ispanic := len(frames) >= 3 && frames[2].Current.Fn != nil && frames[2].Current.Fn.Name == "runtime.gopanic"
|
||||||
|
isdeferreturn := false
|
||||||
|
if len(frames) >= 1 {
|
||||||
|
for _, pc := range bp.DeferReturns {
|
||||||
|
if frames[0].Ret == pc {
|
||||||
|
isdeferreturn = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !ispanic && !isdeferreturn {
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
scope, err := thread.Scope()
|
||||||
|
if err != nil {
|
||||||
|
return true, err
|
||||||
|
}
|
||||||
|
v, err := scope.evalAST(bp.Cond)
|
||||||
|
if err != nil {
|
||||||
|
return true, fmt.Errorf("error evaluating expression: %v", err)
|
||||||
|
}
|
||||||
|
if v.Unreadable != nil {
|
||||||
|
return true, fmt.Errorf("condition expression unreadable: %v", v.Unreadable)
|
||||||
|
}
|
||||||
|
if v.Kind != reflect.Bool {
|
||||||
|
return true, errors.New("condition expression not boolean")
|
||||||
|
}
|
||||||
|
return constant.BoolVal(v.Value), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Internal returns true for breakpoints not set directly by the user.
|
||||||
|
func (bp *Breakpoint) Internal() bool {
|
||||||
|
return bp.Kind != UserBreakpoint
|
||||||
|
}
|
||||||
|
|
||||||
|
// NoBreakpointError is returned when trying to
|
||||||
|
// clear a breakpoint that does not exist.
|
||||||
|
type NoBreakpointError struct {
|
||||||
|
addr uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
func (nbp NoBreakpointError) Error() string {
|
||||||
|
return fmt.Sprintf("no breakpoint at %#v", nbp.addr)
|
||||||
|
}
|
67
vendor/github.com/derekparker/delve/pkg/proc/disasm.go
generated
vendored
Normal file
67
vendor/github.com/derekparker/delve/pkg/proc/disasm.go
generated
vendored
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
type AsmInstruction struct {
|
||||||
|
Loc Location
|
||||||
|
DestLoc *Location
|
||||||
|
Bytes []byte
|
||||||
|
Breakpoint bool
|
||||||
|
AtPC bool
|
||||||
|
Inst *ArchInst
|
||||||
|
}
|
||||||
|
|
||||||
|
type AssemblyFlavour int
|
||||||
|
|
||||||
|
const (
|
||||||
|
GNUFlavour = AssemblyFlavour(iota)
|
||||||
|
IntelFlavour
|
||||||
|
)
|
||||||
|
|
||||||
|
// Disassemble disassembles target memory between startPC and endPC
|
||||||
|
// If currentGoroutine is set and thread is stopped at a CALL instruction Disassemble will evaluate the argument of the CALL instruction using the thread's registers
|
||||||
|
// Be aware that the Bytes field of each returned instruction is a slice of a larger array of size endPC - startPC
|
||||||
|
func (thread *Thread) Disassemble(startPC, endPC uint64, currentGoroutine bool) ([]AsmInstruction, error) {
|
||||||
|
if thread.dbp.exited {
|
||||||
|
return nil, &ProcessExitedError{}
|
||||||
|
}
|
||||||
|
mem, err := thread.readMemory(uintptr(startPC), int(endPC-startPC))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
r := make([]AsmInstruction, 0, len(mem)/15)
|
||||||
|
pc := startPC
|
||||||
|
|
||||||
|
var curpc uint64
|
||||||
|
var regs Registers
|
||||||
|
if currentGoroutine {
|
||||||
|
regs, _ = thread.Registers(false)
|
||||||
|
if regs != nil {
|
||||||
|
curpc = regs.PC()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for len(mem) > 0 {
|
||||||
|
bp, atbp := thread.dbp.breakpoints[pc]
|
||||||
|
if atbp {
|
||||||
|
for i := range bp.OriginalData {
|
||||||
|
mem[i] = bp.OriginalData[i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
file, line, fn := thread.dbp.PCToLine(pc)
|
||||||
|
loc := Location{PC: pc, File: file, Line: line, Fn: fn}
|
||||||
|
inst, err := asmDecode(mem, pc)
|
||||||
|
if err == nil {
|
||||||
|
atpc := currentGoroutine && (curpc == pc)
|
||||||
|
destloc := thread.resolveCallArg(inst, atpc, regs)
|
||||||
|
r = append(r, AsmInstruction{Loc: loc, DestLoc: destloc, Bytes: mem[:inst.Len], Breakpoint: atbp, AtPC: atpc, Inst: inst})
|
||||||
|
|
||||||
|
pc += uint64(inst.Size())
|
||||||
|
mem = mem[inst.Size():]
|
||||||
|
} else {
|
||||||
|
r = append(r, AsmInstruction{Loc: loc, Bytes: mem[:1], Breakpoint: atbp, Inst: nil})
|
||||||
|
pc++
|
||||||
|
mem = mem[1:]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return r, nil
|
||||||
|
}
|
184
vendor/github.com/derekparker/delve/pkg/proc/disasm_amd64.go
generated
vendored
Normal file
184
vendor/github.com/derekparker/delve/pkg/proc/disasm_amd64.go
generated
vendored
Normal file
@ -0,0 +1,184 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"debug/gosym"
|
||||||
|
"encoding/binary"
|
||||||
|
|
||||||
|
"rsc.io/x86/x86asm"
|
||||||
|
)
|
||||||
|
|
||||||
|
var maxInstructionLength uint64 = 15
|
||||||
|
|
||||||
|
type ArchInst x86asm.Inst
|
||||||
|
|
||||||
|
func asmDecode(mem []byte, pc uint64) (*ArchInst, error) {
|
||||||
|
inst, err := x86asm.Decode(mem, 64)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
patchPCRel(pc, &inst)
|
||||||
|
r := ArchInst(inst)
|
||||||
|
return &r, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (inst *ArchInst) Size() int {
|
||||||
|
return inst.Len
|
||||||
|
}
|
||||||
|
|
||||||
|
// converts PC relative arguments to absolute addresses
|
||||||
|
func patchPCRel(pc uint64, inst *x86asm.Inst) {
|
||||||
|
for i := range inst.Args {
|
||||||
|
rel, isrel := inst.Args[i].(x86asm.Rel)
|
||||||
|
if isrel {
|
||||||
|
inst.Args[i] = x86asm.Imm(int64(pc) + int64(rel) + int64(inst.Len))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (inst *AsmInstruction) Text(flavour AssemblyFlavour) string {
|
||||||
|
if inst.Inst == nil {
|
||||||
|
return "?"
|
||||||
|
}
|
||||||
|
|
||||||
|
var text string
|
||||||
|
|
||||||
|
switch flavour {
|
||||||
|
case GNUFlavour:
|
||||||
|
text = x86asm.GNUSyntax(x86asm.Inst(*inst.Inst))
|
||||||
|
case IntelFlavour:
|
||||||
|
fallthrough
|
||||||
|
default:
|
||||||
|
text = x86asm.IntelSyntax(x86asm.Inst(*inst.Inst))
|
||||||
|
}
|
||||||
|
|
||||||
|
if inst.IsCall() && inst.DestLoc != nil && inst.DestLoc.Fn != nil {
|
||||||
|
text += " " + inst.DestLoc.Fn.Name
|
||||||
|
}
|
||||||
|
|
||||||
|
return text
|
||||||
|
}
|
||||||
|
|
||||||
|
func (inst *AsmInstruction) IsCall() bool {
|
||||||
|
return inst.Inst.Op == x86asm.CALL || inst.Inst.Op == x86asm.LCALL
|
||||||
|
}
|
||||||
|
|
||||||
|
func (thread *Thread) resolveCallArg(inst *ArchInst, currentGoroutine bool, regs Registers) *Location {
|
||||||
|
if inst.Op != x86asm.CALL && inst.Op != x86asm.LCALL {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var pc uint64
|
||||||
|
var err error
|
||||||
|
|
||||||
|
switch arg := inst.Args[0].(type) {
|
||||||
|
case x86asm.Imm:
|
||||||
|
pc = uint64(arg)
|
||||||
|
case x86asm.Reg:
|
||||||
|
if !currentGoroutine || regs == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
pc, err = regs.Get(int(arg))
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
case x86asm.Mem:
|
||||||
|
if !currentGoroutine || regs == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if arg.Segment != 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
regs, err := thread.Registers(false)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
base, err1 := regs.Get(int(arg.Base))
|
||||||
|
index, err2 := regs.Get(int(arg.Index))
|
||||||
|
if err1 != nil || err2 != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
addr := uintptr(int64(base) + int64(index*uint64(arg.Scale)) + arg.Disp)
|
||||||
|
//TODO: should this always be 64 bits instead of inst.MemBytes?
|
||||||
|
pcbytes, err := thread.readMemory(addr, inst.MemBytes)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
pc = binary.LittleEndian.Uint64(pcbytes)
|
||||||
|
default:
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
file, line, fn := thread.dbp.PCToLine(pc)
|
||||||
|
if fn == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return &Location{PC: pc, File: file, Line: line, Fn: fn}
|
||||||
|
}
|
||||||
|
|
||||||
|
type instrseq []x86asm.Op
|
||||||
|
|
||||||
|
// Possible stacksplit prologues are inserted by stacksplit in
|
||||||
|
// $GOROOT/src/cmd/internal/obj/x86/obj6.go.
|
||||||
|
// The stacksplit prologue will always begin with loading curg in CX, this
|
||||||
|
// instruction is added by load_g_cx in the same file and is either 1 or 2
|
||||||
|
// MOVs.
|
||||||
|
var prologues []instrseq
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
var tinyStacksplit = instrseq{x86asm.CMP, x86asm.JBE}
|
||||||
|
var smallStacksplit = instrseq{x86asm.LEA, x86asm.CMP, x86asm.JBE}
|
||||||
|
var bigStacksplit = instrseq{x86asm.MOV, x86asm.CMP, x86asm.JE, x86asm.LEA, x86asm.SUB, x86asm.CMP, x86asm.JBE}
|
||||||
|
var unixGetG = instrseq{x86asm.MOV}
|
||||||
|
var windowsGetG = instrseq{x86asm.MOV, x86asm.MOV}
|
||||||
|
|
||||||
|
prologues = make([]instrseq, 0, 2*3)
|
||||||
|
for _, getG := range []instrseq{unixGetG, windowsGetG} {
|
||||||
|
for _, stacksplit := range []instrseq{tinyStacksplit, smallStacksplit, bigStacksplit} {
|
||||||
|
prologue := make(instrseq, 0, len(getG)+len(stacksplit))
|
||||||
|
prologue = append(prologue, getG...)
|
||||||
|
prologue = append(prologue, stacksplit...)
|
||||||
|
prologues = append(prologues, prologue)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FirstPCAfterPrologue returns the address of the first instruction after the prologue for function fn
|
||||||
|
// If sameline is set FirstPCAfterPrologue will always return an address associated with the same line as fn.Entry
|
||||||
|
func (dbp *Process) FirstPCAfterPrologue(fn *gosym.Func, sameline bool) (uint64, error) {
|
||||||
|
text, err := dbp.CurrentThread().Disassemble(fn.Entry, fn.End, false)
|
||||||
|
if err != nil {
|
||||||
|
return fn.Entry, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(text) <= 0 {
|
||||||
|
return fn.Entry, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, prologue := range prologues {
|
||||||
|
if len(prologue) >= len(text) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if checkPrologue(text, prologue) {
|
||||||
|
r := &text[len(prologue)]
|
||||||
|
if sameline {
|
||||||
|
if r.Loc.Line != text[0].Loc.Line {
|
||||||
|
return fn.Entry, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return r.Loc.PC, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return fn.Entry, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func checkPrologue(s []AsmInstruction, prologuePattern instrseq) bool {
|
||||||
|
line := s[0].Loc.Line
|
||||||
|
for i, op := range prologuePattern {
|
||||||
|
if s[i].Inst.Op != op || s[i].Loc.Line != line {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
9
vendor/github.com/derekparker/delve/pkg/proc/doc.go
generated
vendored
Normal file
9
vendor/github.com/derekparker/delve/pkg/proc/doc.go
generated
vendored
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
// Package proc is a low-level package that provides methods to manipulate
|
||||||
|
// the process we are debugging.
|
||||||
|
//
|
||||||
|
// proc implements all core functionality including:
|
||||||
|
// * creating / attaching to a process
|
||||||
|
// * process manipulation (step, next, continue, halt)
|
||||||
|
// * methods to explore the memory of the process
|
||||||
|
//
|
||||||
|
package proc
|
1131
vendor/github.com/derekparker/delve/pkg/proc/eval.go
generated
vendored
Normal file
1131
vendor/github.com/derekparker/delve/pkg/proc/eval.go
generated
vendored
Normal file
File diff suppressed because it is too large
Load Diff
283
vendor/github.com/derekparker/delve/pkg/proc/exc.h
generated
vendored
Normal file
283
vendor/github.com/derekparker/delve/pkg/proc/exc.h
generated
vendored
Normal file
@ -0,0 +1,283 @@
|
|||||||
|
#ifndef _exc_user_
|
||||||
|
#define _exc_user_
|
||||||
|
|
||||||
|
/* Module exc */
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
#include <mach/ndr.h>
|
||||||
|
#include <mach/boolean.h>
|
||||||
|
#include <mach/kern_return.h>
|
||||||
|
#include <mach/notify.h>
|
||||||
|
#include <mach/mach_types.h>
|
||||||
|
#include <mach/message.h>
|
||||||
|
#include <mach/mig_errors.h>
|
||||||
|
#include <mach/port.h>
|
||||||
|
|
||||||
|
/* BEGIN VOUCHER CODE */
|
||||||
|
|
||||||
|
#ifndef KERNEL
|
||||||
|
#if defined(__has_include)
|
||||||
|
#if __has_include(<mach/mig_voucher_support.h>)
|
||||||
|
#ifndef USING_VOUCHERS
|
||||||
|
#define USING_VOUCHERS
|
||||||
|
#endif
|
||||||
|
#ifndef __VOUCHER_FORWARD_TYPE_DECLS__
|
||||||
|
#define __VOUCHER_FORWARD_TYPE_DECLS__
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
extern boolean_t voucher_mach_msg_set(mach_msg_header_t *msg) __attribute__((weak_import));
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#endif // __VOUCHER_FORWARD_TYPE_DECLS__
|
||||||
|
#endif // __has_include(<mach/mach_voucher_types.h>)
|
||||||
|
#endif // __has_include
|
||||||
|
#endif // !KERNEL
|
||||||
|
|
||||||
|
/* END VOUCHER CODE */
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef AUTOTEST
|
||||||
|
#ifndef FUNCTION_PTR_T
|
||||||
|
#define FUNCTION_PTR_T
|
||||||
|
typedef void (*function_ptr_t)(mach_port_t, char *, mach_msg_type_number_t);
|
||||||
|
typedef struct {
|
||||||
|
char *name;
|
||||||
|
function_ptr_t function;
|
||||||
|
} function_table_entry;
|
||||||
|
typedef function_table_entry *function_table_t;
|
||||||
|
#endif /* FUNCTION_PTR_T */
|
||||||
|
#endif /* AUTOTEST */
|
||||||
|
|
||||||
|
#ifndef exc_MSG_COUNT
|
||||||
|
#define exc_MSG_COUNT 3
|
||||||
|
#endif /* exc_MSG_COUNT */
|
||||||
|
|
||||||
|
#include <mach/std_types.h>
|
||||||
|
#include <mach/mig.h>
|
||||||
|
#include <mach/mig.h>
|
||||||
|
#include <mach/mach_types.h>
|
||||||
|
|
||||||
|
#ifdef __BeforeMigUserHeader
|
||||||
|
__BeforeMigUserHeader
|
||||||
|
#endif /* __BeforeMigUserHeader */
|
||||||
|
|
||||||
|
#include <sys/cdefs.h>
|
||||||
|
__BEGIN_DECLS
|
||||||
|
|
||||||
|
|
||||||
|
/* Routine exception_raise */
|
||||||
|
#ifdef mig_external
|
||||||
|
mig_external
|
||||||
|
#else
|
||||||
|
extern
|
||||||
|
#endif /* mig_external */
|
||||||
|
kern_return_t exception_raise
|
||||||
|
(
|
||||||
|
mach_port_t exception_port,
|
||||||
|
mach_port_t thread,
|
||||||
|
mach_port_t task,
|
||||||
|
exception_type_t exception,
|
||||||
|
exception_data_t code,
|
||||||
|
mach_msg_type_number_t codeCnt
|
||||||
|
);
|
||||||
|
|
||||||
|
/* Routine exception_raise_state */
|
||||||
|
#ifdef mig_external
|
||||||
|
mig_external
|
||||||
|
#else
|
||||||
|
extern
|
||||||
|
#endif /* mig_external */
|
||||||
|
kern_return_t exception_raise_state
|
||||||
|
(
|
||||||
|
mach_port_t exception_port,
|
||||||
|
exception_type_t exception,
|
||||||
|
const exception_data_t code,
|
||||||
|
mach_msg_type_number_t codeCnt,
|
||||||
|
int *flavor,
|
||||||
|
const thread_state_t old_state,
|
||||||
|
mach_msg_type_number_t old_stateCnt,
|
||||||
|
thread_state_t new_state,
|
||||||
|
mach_msg_type_number_t *new_stateCnt
|
||||||
|
);
|
||||||
|
|
||||||
|
/* Routine exception_raise_state_identity */
|
||||||
|
#ifdef mig_external
|
||||||
|
mig_external
|
||||||
|
#else
|
||||||
|
extern
|
||||||
|
#endif /* mig_external */
|
||||||
|
kern_return_t exception_raise_state_identity
|
||||||
|
(
|
||||||
|
mach_port_t exception_port,
|
||||||
|
mach_port_t thread,
|
||||||
|
mach_port_t task,
|
||||||
|
exception_type_t exception,
|
||||||
|
exception_data_t code,
|
||||||
|
mach_msg_type_number_t codeCnt,
|
||||||
|
int *flavor,
|
||||||
|
thread_state_t old_state,
|
||||||
|
mach_msg_type_number_t old_stateCnt,
|
||||||
|
thread_state_t new_state,
|
||||||
|
mach_msg_type_number_t *new_stateCnt
|
||||||
|
);
|
||||||
|
|
||||||
|
__END_DECLS
|
||||||
|
|
||||||
|
/********************** Caution **************************/
|
||||||
|
/* The following data types should be used to calculate */
|
||||||
|
/* maximum message sizes only. The actual message may be */
|
||||||
|
/* smaller, and the position of the arguments within the */
|
||||||
|
/* message layout may vary from what is presented here. */
|
||||||
|
/* For example, if any of the arguments are variable- */
|
||||||
|
/* sized, and less than the maximum is sent, the data */
|
||||||
|
/* will be packed tight in the actual message to reduce */
|
||||||
|
/* the presence of holes. */
|
||||||
|
/********************** Caution **************************/
|
||||||
|
|
||||||
|
/* typedefs for all requests */
|
||||||
|
|
||||||
|
#ifndef __Request__exc_subsystem__defined
|
||||||
|
#define __Request__exc_subsystem__defined
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
/* start of the kernel processed data */
|
||||||
|
mach_msg_body_t msgh_body;
|
||||||
|
mach_msg_port_descriptor_t thread;
|
||||||
|
mach_msg_port_descriptor_t task;
|
||||||
|
/* end of the kernel processed data */
|
||||||
|
NDR_record_t NDR;
|
||||||
|
exception_type_t exception;
|
||||||
|
mach_msg_type_number_t codeCnt;
|
||||||
|
integer_t code[2];
|
||||||
|
} __Request__exception_raise_t;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
exception_type_t exception;
|
||||||
|
mach_msg_type_number_t codeCnt;
|
||||||
|
integer_t code[2];
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t old_stateCnt;
|
||||||
|
natural_t old_state[224];
|
||||||
|
} __Request__exception_raise_state_t;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
/* start of the kernel processed data */
|
||||||
|
mach_msg_body_t msgh_body;
|
||||||
|
mach_msg_port_descriptor_t thread;
|
||||||
|
mach_msg_port_descriptor_t task;
|
||||||
|
/* end of the kernel processed data */
|
||||||
|
NDR_record_t NDR;
|
||||||
|
exception_type_t exception;
|
||||||
|
mach_msg_type_number_t codeCnt;
|
||||||
|
integer_t code[2];
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t old_stateCnt;
|
||||||
|
natural_t old_state[224];
|
||||||
|
} __Request__exception_raise_state_identity_t;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
#endif /* !__Request__exc_subsystem__defined */
|
||||||
|
|
||||||
|
/* union of all requests */
|
||||||
|
|
||||||
|
#ifndef __RequestUnion__exc_subsystem__defined
|
||||||
|
#define __RequestUnion__exc_subsystem__defined
|
||||||
|
union __RequestUnion__exc_subsystem {
|
||||||
|
__Request__exception_raise_t Request_exception_raise;
|
||||||
|
__Request__exception_raise_state_t Request_exception_raise_state;
|
||||||
|
__Request__exception_raise_state_identity_t Request_exception_raise_state_identity;
|
||||||
|
};
|
||||||
|
#endif /* !__RequestUnion__exc_subsystem__defined */
|
||||||
|
/* typedefs for all replies */
|
||||||
|
|
||||||
|
#ifndef __Reply__exc_subsystem__defined
|
||||||
|
#define __Reply__exc_subsystem__defined
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
} __Reply__exception_raise_t;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t new_stateCnt;
|
||||||
|
natural_t new_state[224];
|
||||||
|
} __Reply__exception_raise_state_t;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t new_stateCnt;
|
||||||
|
natural_t new_state[224];
|
||||||
|
} __Reply__exception_raise_state_identity_t;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
#endif /* !__Reply__exc_subsystem__defined */
|
||||||
|
|
||||||
|
/* union of all replies */
|
||||||
|
|
||||||
|
#ifndef __ReplyUnion__exc_subsystem__defined
|
||||||
|
#define __ReplyUnion__exc_subsystem__defined
|
||||||
|
union __ReplyUnion__exc_subsystem {
|
||||||
|
__Reply__exception_raise_t Reply_exception_raise;
|
||||||
|
__Reply__exception_raise_state_t Reply_exception_raise_state;
|
||||||
|
__Reply__exception_raise_state_identity_t Reply_exception_raise_state_identity;
|
||||||
|
};
|
||||||
|
#endif /* !__RequestUnion__exc_subsystem__defined */
|
||||||
|
|
||||||
|
#ifndef subsystem_to_name_map_exc
|
||||||
|
#define subsystem_to_name_map_exc \
|
||||||
|
{ "exception_raise", 2401 },\
|
||||||
|
{ "exception_raise_state", 2402 },\
|
||||||
|
{ "exception_raise_state_identity", 2403 }
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __AfterMigUserHeader
|
||||||
|
__AfterMigUserHeader
|
||||||
|
#endif /* __AfterMigUserHeader */
|
||||||
|
|
||||||
|
#endif /* _exc_user_ */
|
768
vendor/github.com/derekparker/delve/pkg/proc/exc_user_darwin.c
generated
vendored
Normal file
768
vendor/github.com/derekparker/delve/pkg/proc/exc_user_darwin.c
generated
vendored
Normal file
@ -0,0 +1,768 @@
|
|||||||
|
/*
|
||||||
|
* IDENTIFICATION:
|
||||||
|
* stub generated Sun Feb 22 20:54:31 2015
|
||||||
|
* with a MiG generated by bootstrap_cmds-91
|
||||||
|
* OPTIONS:
|
||||||
|
*/
|
||||||
|
#define __MIG_check__Reply__exc_subsystem__ 1
|
||||||
|
|
||||||
|
#include "exc.h"
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef mig_internal
|
||||||
|
#define mig_internal static __inline__
|
||||||
|
#endif /* mig_internal */
|
||||||
|
|
||||||
|
#ifndef mig_external
|
||||||
|
#define mig_external
|
||||||
|
#endif /* mig_external */
|
||||||
|
|
||||||
|
#if !defined(__MigTypeCheck) && defined(TypeCheck)
|
||||||
|
#define __MigTypeCheck TypeCheck /* Legacy setting */
|
||||||
|
#endif /* !defined(__MigTypeCheck) */
|
||||||
|
|
||||||
|
#if !defined(__MigKernelSpecificCode) && defined(_MIG_KERNEL_SPECIFIC_CODE_)
|
||||||
|
#define __MigKernelSpecificCode _MIG_KERNEL_SPECIFIC_CODE_ /* Legacy setting */
|
||||||
|
#endif /* !defined(__MigKernelSpecificCode) */
|
||||||
|
|
||||||
|
#ifndef LimitCheck
|
||||||
|
#define LimitCheck 0
|
||||||
|
#endif /* LimitCheck */
|
||||||
|
|
||||||
|
#ifndef min
|
||||||
|
#define min(a,b) ( ((a) < (b))? (a): (b) )
|
||||||
|
#endif /* min */
|
||||||
|
|
||||||
|
#if !defined(_WALIGN_)
|
||||||
|
#define _WALIGN_(x) (((x) + 3) & ~3)
|
||||||
|
#endif /* !defined(_WALIGN_) */
|
||||||
|
|
||||||
|
#if !defined(_WALIGNSZ_)
|
||||||
|
#define _WALIGNSZ_(x) _WALIGN_(sizeof(x))
|
||||||
|
#endif /* !defined(_WALIGNSZ_) */
|
||||||
|
|
||||||
|
#ifndef UseStaticTemplates
|
||||||
|
#define UseStaticTemplates 0
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
#ifndef __MachMsgErrorWithTimeout
|
||||||
|
#define __MachMsgErrorWithTimeout(_R_) { \
|
||||||
|
switch (_R_) { \
|
||||||
|
case MACH_SEND_INVALID_DATA: \
|
||||||
|
case MACH_SEND_INVALID_DEST: \
|
||||||
|
case MACH_SEND_INVALID_HEADER: \
|
||||||
|
mig_put_reply_port(InP->Head.msgh_reply_port); \
|
||||||
|
break; \
|
||||||
|
case MACH_SEND_TIMED_OUT: \
|
||||||
|
case MACH_RCV_TIMED_OUT: \
|
||||||
|
default: \
|
||||||
|
mig_dealloc_reply_port(InP->Head.msgh_reply_port); \
|
||||||
|
} \
|
||||||
|
}
|
||||||
|
#endif /* __MachMsgErrorWithTimeout */
|
||||||
|
|
||||||
|
#ifndef __MachMsgErrorWithoutTimeout
|
||||||
|
#define __MachMsgErrorWithoutTimeout(_R_) { \
|
||||||
|
switch (_R_) { \
|
||||||
|
case MACH_SEND_INVALID_DATA: \
|
||||||
|
case MACH_SEND_INVALID_DEST: \
|
||||||
|
case MACH_SEND_INVALID_HEADER: \
|
||||||
|
mig_put_reply_port(InP->Head.msgh_reply_port); \
|
||||||
|
break; \
|
||||||
|
default: \
|
||||||
|
mig_dealloc_reply_port(InP->Head.msgh_reply_port); \
|
||||||
|
} \
|
||||||
|
}
|
||||||
|
#endif /* __MachMsgErrorWithoutTimeout */
|
||||||
|
|
||||||
|
#ifndef __DeclareSendRpc
|
||||||
|
#define __DeclareSendRpc(_NUM_, _NAME_)
|
||||||
|
#endif /* __DeclareSendRpc */
|
||||||
|
|
||||||
|
#ifndef __BeforeSendRpc
|
||||||
|
#define __BeforeSendRpc(_NUM_, _NAME_)
|
||||||
|
#endif /* __BeforeSendRpc */
|
||||||
|
|
||||||
|
#ifndef __AfterSendRpc
|
||||||
|
#define __AfterSendRpc(_NUM_, _NAME_)
|
||||||
|
#endif /* __AfterSendRpc */
|
||||||
|
|
||||||
|
#ifndef __DeclareSendSimple
|
||||||
|
#define __DeclareSendSimple(_NUM_, _NAME_)
|
||||||
|
#endif /* __DeclareSendSimple */
|
||||||
|
|
||||||
|
#ifndef __BeforeSendSimple
|
||||||
|
#define __BeforeSendSimple(_NUM_, _NAME_)
|
||||||
|
#endif /* __BeforeSendSimple */
|
||||||
|
|
||||||
|
#ifndef __AfterSendSimple
|
||||||
|
#define __AfterSendSimple(_NUM_, _NAME_)
|
||||||
|
#endif /* __AfterSendSimple */
|
||||||
|
|
||||||
|
#define msgh_request_port msgh_remote_port
|
||||||
|
#define msgh_reply_port msgh_local_port
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#if ( __MigTypeCheck )
|
||||||
|
#if __MIG_check__Reply__exc_subsystem__
|
||||||
|
#if !defined(__MIG_check__Reply__exception_raise_t__defined)
|
||||||
|
#define __MIG_check__Reply__exception_raise_t__defined
|
||||||
|
|
||||||
|
mig_internal kern_return_t __MIG_check__Reply__exception_raise_t(__Reply__exception_raise_t *Out0P)
|
||||||
|
{
|
||||||
|
|
||||||
|
typedef __Reply__exception_raise_t __Reply;
|
||||||
|
if (Out0P->Head.msgh_id != 2501) {
|
||||||
|
if (Out0P->Head.msgh_id == MACH_NOTIFY_SEND_ONCE)
|
||||||
|
{ return MIG_SERVER_DIED; }
|
||||||
|
else
|
||||||
|
{ return MIG_REPLY_MISMATCH; }
|
||||||
|
}
|
||||||
|
|
||||||
|
#if __MigTypeCheck
|
||||||
|
if ((Out0P->Head.msgh_bits & MACH_MSGH_BITS_COMPLEX) ||
|
||||||
|
(Out0P->Head.msgh_size != (mach_msg_size_t)sizeof(__Reply)))
|
||||||
|
{ return MIG_TYPE_ERROR ; }
|
||||||
|
#endif /* __MigTypeCheck */
|
||||||
|
|
||||||
|
{
|
||||||
|
return Out0P->RetCode;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif /* !defined(__MIG_check__Reply__exception_raise_t__defined) */
|
||||||
|
#endif /* __MIG_check__Reply__exc_subsystem__ */
|
||||||
|
#endif /* ( __MigTypeCheck ) */
|
||||||
|
|
||||||
|
|
||||||
|
/* Routine exception_raise */
|
||||||
|
mig_external kern_return_t exception_raise
|
||||||
|
(
|
||||||
|
mach_port_t exception_port,
|
||||||
|
mach_port_t thread,
|
||||||
|
mach_port_t task,
|
||||||
|
exception_type_t exception,
|
||||||
|
exception_data_t code,
|
||||||
|
mach_msg_type_number_t codeCnt
|
||||||
|
)
|
||||||
|
{
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
/* start of the kernel processed data */
|
||||||
|
mach_msg_body_t msgh_body;
|
||||||
|
mach_msg_port_descriptor_t thread;
|
||||||
|
mach_msg_port_descriptor_t task;
|
||||||
|
/* end of the kernel processed data */
|
||||||
|
NDR_record_t NDR;
|
||||||
|
exception_type_t exception;
|
||||||
|
mach_msg_type_number_t codeCnt;
|
||||||
|
integer_t code[2];
|
||||||
|
} Request;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
mach_msg_trailer_t trailer;
|
||||||
|
} Reply;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
} __Reply;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
/*
|
||||||
|
* typedef struct {
|
||||||
|
* mach_msg_header_t Head;
|
||||||
|
* NDR_record_t NDR;
|
||||||
|
* kern_return_t RetCode;
|
||||||
|
* } mig_reply_error_t;
|
||||||
|
*/
|
||||||
|
|
||||||
|
union {
|
||||||
|
Request In;
|
||||||
|
Reply Out;
|
||||||
|
} Mess;
|
||||||
|
|
||||||
|
Request *InP = &Mess.In;
|
||||||
|
Reply *Out0P = &Mess.Out;
|
||||||
|
|
||||||
|
mach_msg_return_t msg_result;
|
||||||
|
unsigned int msgh_size;
|
||||||
|
|
||||||
|
#ifdef __MIG_check__Reply__exception_raise_t__defined
|
||||||
|
kern_return_t check_result;
|
||||||
|
#endif /* __MIG_check__Reply__exception_raise_t__defined */
|
||||||
|
|
||||||
|
__DeclareSendRpc(2401, "exception_raise")
|
||||||
|
|
||||||
|
#if UseStaticTemplates
|
||||||
|
const static mach_msg_port_descriptor_t threadTemplate = {
|
||||||
|
/* name = */ MACH_PORT_NULL,
|
||||||
|
/* pad1 = */ 0,
|
||||||
|
/* pad2 = */ 0,
|
||||||
|
/* disp = */ 19,
|
||||||
|
/* type = */ MACH_MSG_PORT_DESCRIPTOR,
|
||||||
|
};
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
#if UseStaticTemplates
|
||||||
|
const static mach_msg_port_descriptor_t taskTemplate = {
|
||||||
|
/* name = */ MACH_PORT_NULL,
|
||||||
|
/* pad1 = */ 0,
|
||||||
|
/* pad2 = */ 0,
|
||||||
|
/* disp = */ 19,
|
||||||
|
/* type = */ MACH_MSG_PORT_DESCRIPTOR,
|
||||||
|
};
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
InP->msgh_body.msgh_descriptor_count = 2;
|
||||||
|
#if UseStaticTemplates
|
||||||
|
InP->thread = threadTemplate;
|
||||||
|
InP->thread.name = thread;
|
||||||
|
#else /* UseStaticTemplates */
|
||||||
|
InP->thread.name = thread;
|
||||||
|
InP->thread.disposition = 19;
|
||||||
|
InP->thread.type = MACH_MSG_PORT_DESCRIPTOR;
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
#if UseStaticTemplates
|
||||||
|
InP->task = taskTemplate;
|
||||||
|
InP->task.name = task;
|
||||||
|
#else /* UseStaticTemplates */
|
||||||
|
InP->task.name = task;
|
||||||
|
InP->task.disposition = 19;
|
||||||
|
InP->task.type = MACH_MSG_PORT_DESCRIPTOR;
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
InP->NDR = NDR_record;
|
||||||
|
|
||||||
|
InP->exception = exception;
|
||||||
|
|
||||||
|
if (codeCnt > 2) {
|
||||||
|
{ return MIG_ARRAY_TOO_LARGE; }
|
||||||
|
}
|
||||||
|
(void)memcpy((char *) InP->code, (const char *) code, 4 * codeCnt);
|
||||||
|
|
||||||
|
InP->codeCnt = codeCnt;
|
||||||
|
|
||||||
|
msgh_size = (mach_msg_size_t)(sizeof(Request) - 8) + ((4 * codeCnt));
|
||||||
|
InP->Head.msgh_bits = MACH_MSGH_BITS_COMPLEX|
|
||||||
|
MACH_MSGH_BITS(19, MACH_MSG_TYPE_MAKE_SEND_ONCE);
|
||||||
|
/* msgh_size passed as argument */
|
||||||
|
InP->Head.msgh_request_port = exception_port;
|
||||||
|
InP->Head.msgh_reply_port = mig_get_reply_port();
|
||||||
|
InP->Head.msgh_id = 2401;
|
||||||
|
|
||||||
|
/* BEGIN VOUCHER CODE */
|
||||||
|
|
||||||
|
#ifdef USING_VOUCHERS
|
||||||
|
if (voucher_mach_msg_set != NULL) {
|
||||||
|
voucher_mach_msg_set(&InP->Head);
|
||||||
|
}
|
||||||
|
#endif // USING_VOUCHERS
|
||||||
|
|
||||||
|
/* END VOUCHER CODE */
|
||||||
|
|
||||||
|
__BeforeSendRpc(2401, "exception_raise")
|
||||||
|
msg_result = mach_msg(&InP->Head, MACH_SEND_MSG|MACH_RCV_MSG|MACH_MSG_OPTION_NONE, msgh_size, (mach_msg_size_t)sizeof(Reply), InP->Head.msgh_reply_port, MACH_MSG_TIMEOUT_NONE, MACH_PORT_NULL);
|
||||||
|
__AfterSendRpc(2401, "exception_raise")
|
||||||
|
if (msg_result != MACH_MSG_SUCCESS) {
|
||||||
|
__MachMsgErrorWithoutTimeout(msg_result);
|
||||||
|
{ return msg_result; }
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#if defined(__MIG_check__Reply__exception_raise_t__defined)
|
||||||
|
check_result = __MIG_check__Reply__exception_raise_t((__Reply__exception_raise_t *)Out0P);
|
||||||
|
if (check_result != MACH_MSG_SUCCESS)
|
||||||
|
{ return check_result; }
|
||||||
|
#endif /* defined(__MIG_check__Reply__exception_raise_t__defined) */
|
||||||
|
|
||||||
|
return KERN_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if ( __MigTypeCheck )
|
||||||
|
#if __MIG_check__Reply__exc_subsystem__
|
||||||
|
#if !defined(__MIG_check__Reply__exception_raise_state_t__defined)
|
||||||
|
#define __MIG_check__Reply__exception_raise_state_t__defined
|
||||||
|
|
||||||
|
mig_internal kern_return_t __MIG_check__Reply__exception_raise_state_t(__Reply__exception_raise_state_t *Out0P)
|
||||||
|
{
|
||||||
|
|
||||||
|
typedef __Reply__exception_raise_state_t __Reply;
|
||||||
|
#if __MigTypeCheck
|
||||||
|
unsigned int msgh_size;
|
||||||
|
#endif /* __MigTypeCheck */
|
||||||
|
|
||||||
|
if (Out0P->Head.msgh_id != 2502) {
|
||||||
|
if (Out0P->Head.msgh_id == MACH_NOTIFY_SEND_ONCE)
|
||||||
|
{ return MIG_SERVER_DIED; }
|
||||||
|
else
|
||||||
|
{ return MIG_REPLY_MISMATCH; }
|
||||||
|
}
|
||||||
|
|
||||||
|
#if __MigTypeCheck
|
||||||
|
msgh_size = Out0P->Head.msgh_size;
|
||||||
|
|
||||||
|
if ((Out0P->Head.msgh_bits & MACH_MSGH_BITS_COMPLEX) ||
|
||||||
|
((msgh_size > (mach_msg_size_t)sizeof(__Reply) || msgh_size < (mach_msg_size_t)(sizeof(__Reply) - 896)) &&
|
||||||
|
(msgh_size != (mach_msg_size_t)sizeof(mig_reply_error_t) ||
|
||||||
|
Out0P->RetCode == KERN_SUCCESS)))
|
||||||
|
{ return MIG_TYPE_ERROR ; }
|
||||||
|
#endif /* __MigTypeCheck */
|
||||||
|
|
||||||
|
if (Out0P->RetCode != KERN_SUCCESS) {
|
||||||
|
return ((mig_reply_error_t *)Out0P)->RetCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if __MigTypeCheck
|
||||||
|
if ( Out0P->new_stateCnt > 224 )
|
||||||
|
return MIG_TYPE_ERROR;
|
||||||
|
if (((msgh_size - (mach_msg_size_t)(sizeof(__Reply) - 896)) / 4< Out0P->new_stateCnt) ||
|
||||||
|
(msgh_size != (mach_msg_size_t)(sizeof(__Reply) - 896) + Out0P->new_stateCnt * 4))
|
||||||
|
{ return MIG_TYPE_ERROR ; }
|
||||||
|
#endif /* __MigTypeCheck */
|
||||||
|
|
||||||
|
return MACH_MSG_SUCCESS;
|
||||||
|
}
|
||||||
|
#endif /* !defined(__MIG_check__Reply__exception_raise_state_t__defined) */
|
||||||
|
#endif /* __MIG_check__Reply__exc_subsystem__ */
|
||||||
|
#endif /* ( __MigTypeCheck ) */
|
||||||
|
|
||||||
|
|
||||||
|
/* Routine exception_raise_state */
|
||||||
|
mig_external kern_return_t exception_raise_state
|
||||||
|
(
|
||||||
|
mach_port_t exception_port,
|
||||||
|
exception_type_t exception,
|
||||||
|
const exception_data_t code,
|
||||||
|
mach_msg_type_number_t codeCnt,
|
||||||
|
int *flavor,
|
||||||
|
const thread_state_t old_state,
|
||||||
|
mach_msg_type_number_t old_stateCnt,
|
||||||
|
thread_state_t new_state,
|
||||||
|
mach_msg_type_number_t *new_stateCnt
|
||||||
|
)
|
||||||
|
{
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
exception_type_t exception;
|
||||||
|
mach_msg_type_number_t codeCnt;
|
||||||
|
integer_t code[2];
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t old_stateCnt;
|
||||||
|
natural_t old_state[224];
|
||||||
|
} Request;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t new_stateCnt;
|
||||||
|
natural_t new_state[224];
|
||||||
|
mach_msg_trailer_t trailer;
|
||||||
|
} Reply;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t new_stateCnt;
|
||||||
|
natural_t new_state[224];
|
||||||
|
} __Reply;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
/*
|
||||||
|
* typedef struct {
|
||||||
|
* mach_msg_header_t Head;
|
||||||
|
* NDR_record_t NDR;
|
||||||
|
* kern_return_t RetCode;
|
||||||
|
* } mig_reply_error_t;
|
||||||
|
*/
|
||||||
|
|
||||||
|
union {
|
||||||
|
Request In;
|
||||||
|
Reply Out;
|
||||||
|
} Mess;
|
||||||
|
|
||||||
|
Request *InP = &Mess.In;
|
||||||
|
Reply *Out0P = &Mess.Out;
|
||||||
|
|
||||||
|
mach_msg_return_t msg_result;
|
||||||
|
unsigned int msgh_size;
|
||||||
|
unsigned int msgh_size_delta;
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef __MIG_check__Reply__exception_raise_state_t__defined
|
||||||
|
kern_return_t check_result;
|
||||||
|
#endif /* __MIG_check__Reply__exception_raise_state_t__defined */
|
||||||
|
|
||||||
|
__DeclareSendRpc(2402, "exception_raise_state")
|
||||||
|
|
||||||
|
InP->NDR = NDR_record;
|
||||||
|
|
||||||
|
InP->exception = exception;
|
||||||
|
|
||||||
|
if (codeCnt > 2) {
|
||||||
|
{ return MIG_ARRAY_TOO_LARGE; }
|
||||||
|
}
|
||||||
|
(void)memcpy((char *) InP->code, (const char *) code, 4 * codeCnt);
|
||||||
|
|
||||||
|
InP->codeCnt = codeCnt;
|
||||||
|
|
||||||
|
msgh_size_delta = (4 * codeCnt);
|
||||||
|
msgh_size = (mach_msg_size_t)(sizeof(Request) - 904) + msgh_size_delta;
|
||||||
|
InP = (Request *) ((pointer_t) InP + msgh_size_delta - 8);
|
||||||
|
|
||||||
|
InP->flavor = *flavor;
|
||||||
|
|
||||||
|
if (old_stateCnt > 224) {
|
||||||
|
{ return MIG_ARRAY_TOO_LARGE; }
|
||||||
|
}
|
||||||
|
(void)memcpy((char *) InP->old_state, (const char *) old_state, 4 * old_stateCnt);
|
||||||
|
|
||||||
|
InP->old_stateCnt = old_stateCnt;
|
||||||
|
|
||||||
|
msgh_size += (4 * old_stateCnt);
|
||||||
|
InP = &Mess.In;
|
||||||
|
InP->Head.msgh_bits =
|
||||||
|
MACH_MSGH_BITS(19, MACH_MSG_TYPE_MAKE_SEND_ONCE);
|
||||||
|
/* msgh_size passed as argument */
|
||||||
|
InP->Head.msgh_request_port = exception_port;
|
||||||
|
InP->Head.msgh_reply_port = mig_get_reply_port();
|
||||||
|
InP->Head.msgh_id = 2402;
|
||||||
|
|
||||||
|
/* BEGIN VOUCHER CODE */
|
||||||
|
|
||||||
|
#ifdef USING_VOUCHERS
|
||||||
|
if (voucher_mach_msg_set != NULL) {
|
||||||
|
voucher_mach_msg_set(&InP->Head);
|
||||||
|
}
|
||||||
|
#endif // USING_VOUCHERS
|
||||||
|
|
||||||
|
/* END VOUCHER CODE */
|
||||||
|
|
||||||
|
__BeforeSendRpc(2402, "exception_raise_state")
|
||||||
|
msg_result = mach_msg(&InP->Head, MACH_SEND_MSG|MACH_RCV_MSG|MACH_MSG_OPTION_NONE, msgh_size, (mach_msg_size_t)sizeof(Reply), InP->Head.msgh_reply_port, MACH_MSG_TIMEOUT_NONE, MACH_PORT_NULL);
|
||||||
|
__AfterSendRpc(2402, "exception_raise_state")
|
||||||
|
if (msg_result != MACH_MSG_SUCCESS) {
|
||||||
|
__MachMsgErrorWithoutTimeout(msg_result);
|
||||||
|
{ return msg_result; }
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#if defined(__MIG_check__Reply__exception_raise_state_t__defined)
|
||||||
|
check_result = __MIG_check__Reply__exception_raise_state_t((__Reply__exception_raise_state_t *)Out0P);
|
||||||
|
if (check_result != MACH_MSG_SUCCESS)
|
||||||
|
{ return check_result; }
|
||||||
|
#endif /* defined(__MIG_check__Reply__exception_raise_state_t__defined) */
|
||||||
|
|
||||||
|
*flavor = Out0P->flavor;
|
||||||
|
|
||||||
|
if (Out0P->new_stateCnt > 224) {
|
||||||
|
(void)memcpy((char *) new_state, (const char *) Out0P->new_state, 4 * 224);
|
||||||
|
*new_stateCnt = Out0P->new_stateCnt;
|
||||||
|
{ return MIG_ARRAY_TOO_LARGE; }
|
||||||
|
}
|
||||||
|
(void)memcpy((char *) new_state, (const char *) Out0P->new_state, 4 * Out0P->new_stateCnt);
|
||||||
|
|
||||||
|
*new_stateCnt = Out0P->new_stateCnt;
|
||||||
|
|
||||||
|
return KERN_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if ( __MigTypeCheck )
|
||||||
|
#if __MIG_check__Reply__exc_subsystem__
|
||||||
|
#if !defined(__MIG_check__Reply__exception_raise_state_identity_t__defined)
|
||||||
|
#define __MIG_check__Reply__exception_raise_state_identity_t__defined
|
||||||
|
|
||||||
|
mig_internal kern_return_t __MIG_check__Reply__exception_raise_state_identity_t(__Reply__exception_raise_state_identity_t *Out0P)
|
||||||
|
{
|
||||||
|
|
||||||
|
typedef __Reply__exception_raise_state_identity_t __Reply;
|
||||||
|
#if __MigTypeCheck
|
||||||
|
unsigned int msgh_size;
|
||||||
|
#endif /* __MigTypeCheck */
|
||||||
|
|
||||||
|
if (Out0P->Head.msgh_id != 2503) {
|
||||||
|
if (Out0P->Head.msgh_id == MACH_NOTIFY_SEND_ONCE)
|
||||||
|
{ return MIG_SERVER_DIED; }
|
||||||
|
else
|
||||||
|
{ return MIG_REPLY_MISMATCH; }
|
||||||
|
}
|
||||||
|
|
||||||
|
#if __MigTypeCheck
|
||||||
|
msgh_size = Out0P->Head.msgh_size;
|
||||||
|
|
||||||
|
if ((Out0P->Head.msgh_bits & MACH_MSGH_BITS_COMPLEX) ||
|
||||||
|
((msgh_size > (mach_msg_size_t)sizeof(__Reply) || msgh_size < (mach_msg_size_t)(sizeof(__Reply) - 896)) &&
|
||||||
|
(msgh_size != (mach_msg_size_t)sizeof(mig_reply_error_t) ||
|
||||||
|
Out0P->RetCode == KERN_SUCCESS)))
|
||||||
|
{ return MIG_TYPE_ERROR ; }
|
||||||
|
#endif /* __MigTypeCheck */
|
||||||
|
|
||||||
|
if (Out0P->RetCode != KERN_SUCCESS) {
|
||||||
|
return ((mig_reply_error_t *)Out0P)->RetCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if __MigTypeCheck
|
||||||
|
if ( Out0P->new_stateCnt > 224 )
|
||||||
|
return MIG_TYPE_ERROR;
|
||||||
|
if (((msgh_size - (mach_msg_size_t)(sizeof(__Reply) - 896)) / 4< Out0P->new_stateCnt) ||
|
||||||
|
(msgh_size != (mach_msg_size_t)(sizeof(__Reply) - 896) + Out0P->new_stateCnt * 4))
|
||||||
|
{ return MIG_TYPE_ERROR ; }
|
||||||
|
#endif /* __MigTypeCheck */
|
||||||
|
|
||||||
|
return MACH_MSG_SUCCESS;
|
||||||
|
}
|
||||||
|
#endif /* !defined(__MIG_check__Reply__exception_raise_state_identity_t__defined) */
|
||||||
|
#endif /* __MIG_check__Reply__exc_subsystem__ */
|
||||||
|
#endif /* ( __MigTypeCheck ) */
|
||||||
|
|
||||||
|
|
||||||
|
/* Routine exception_raise_state_identity */
|
||||||
|
mig_external kern_return_t exception_raise_state_identity
|
||||||
|
(
|
||||||
|
mach_port_t exception_port,
|
||||||
|
mach_port_t thread,
|
||||||
|
mach_port_t task,
|
||||||
|
exception_type_t exception,
|
||||||
|
exception_data_t code,
|
||||||
|
mach_msg_type_number_t codeCnt,
|
||||||
|
int *flavor,
|
||||||
|
thread_state_t old_state,
|
||||||
|
mach_msg_type_number_t old_stateCnt,
|
||||||
|
thread_state_t new_state,
|
||||||
|
mach_msg_type_number_t *new_stateCnt
|
||||||
|
)
|
||||||
|
{
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
/* start of the kernel processed data */
|
||||||
|
mach_msg_body_t msgh_body;
|
||||||
|
mach_msg_port_descriptor_t thread;
|
||||||
|
mach_msg_port_descriptor_t task;
|
||||||
|
/* end of the kernel processed data */
|
||||||
|
NDR_record_t NDR;
|
||||||
|
exception_type_t exception;
|
||||||
|
mach_msg_type_number_t codeCnt;
|
||||||
|
integer_t code[2];
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t old_stateCnt;
|
||||||
|
natural_t old_state[224];
|
||||||
|
} Request;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t new_stateCnt;
|
||||||
|
natural_t new_state[224];
|
||||||
|
mach_msg_trailer_t trailer;
|
||||||
|
} Reply;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t new_stateCnt;
|
||||||
|
natural_t new_state[224];
|
||||||
|
} __Reply;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
/*
|
||||||
|
* typedef struct {
|
||||||
|
* mach_msg_header_t Head;
|
||||||
|
* NDR_record_t NDR;
|
||||||
|
* kern_return_t RetCode;
|
||||||
|
* } mig_reply_error_t;
|
||||||
|
*/
|
||||||
|
|
||||||
|
union {
|
||||||
|
Request In;
|
||||||
|
Reply Out;
|
||||||
|
} Mess;
|
||||||
|
|
||||||
|
Request *InP = &Mess.In;
|
||||||
|
Reply *Out0P = &Mess.Out;
|
||||||
|
|
||||||
|
mach_msg_return_t msg_result;
|
||||||
|
unsigned int msgh_size;
|
||||||
|
unsigned int msgh_size_delta;
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef __MIG_check__Reply__exception_raise_state_identity_t__defined
|
||||||
|
kern_return_t check_result;
|
||||||
|
#endif /* __MIG_check__Reply__exception_raise_state_identity_t__defined */
|
||||||
|
|
||||||
|
__DeclareSendRpc(2403, "exception_raise_state_identity")
|
||||||
|
|
||||||
|
#if UseStaticTemplates
|
||||||
|
const static mach_msg_port_descriptor_t threadTemplate = {
|
||||||
|
/* name = */ MACH_PORT_NULL,
|
||||||
|
/* pad1 = */ 0,
|
||||||
|
/* pad2 = */ 0,
|
||||||
|
/* disp = */ 19,
|
||||||
|
/* type = */ MACH_MSG_PORT_DESCRIPTOR,
|
||||||
|
};
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
#if UseStaticTemplates
|
||||||
|
const static mach_msg_port_descriptor_t taskTemplate = {
|
||||||
|
/* name = */ MACH_PORT_NULL,
|
||||||
|
/* pad1 = */ 0,
|
||||||
|
/* pad2 = */ 0,
|
||||||
|
/* disp = */ 19,
|
||||||
|
/* type = */ MACH_MSG_PORT_DESCRIPTOR,
|
||||||
|
};
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
InP->msgh_body.msgh_descriptor_count = 2;
|
||||||
|
#if UseStaticTemplates
|
||||||
|
InP->thread = threadTemplate;
|
||||||
|
InP->thread.name = thread;
|
||||||
|
#else /* UseStaticTemplates */
|
||||||
|
InP->thread.name = thread;
|
||||||
|
InP->thread.disposition = 19;
|
||||||
|
InP->thread.type = MACH_MSG_PORT_DESCRIPTOR;
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
#if UseStaticTemplates
|
||||||
|
InP->task = taskTemplate;
|
||||||
|
InP->task.name = task;
|
||||||
|
#else /* UseStaticTemplates */
|
||||||
|
InP->task.name = task;
|
||||||
|
InP->task.disposition = 19;
|
||||||
|
InP->task.type = MACH_MSG_PORT_DESCRIPTOR;
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
InP->NDR = NDR_record;
|
||||||
|
|
||||||
|
InP->exception = exception;
|
||||||
|
|
||||||
|
if (codeCnt > 2) {
|
||||||
|
{ return MIG_ARRAY_TOO_LARGE; }
|
||||||
|
}
|
||||||
|
(void)memcpy((char *) InP->code, (const char *) code, 4 * codeCnt);
|
||||||
|
|
||||||
|
InP->codeCnt = codeCnt;
|
||||||
|
|
||||||
|
msgh_size_delta = (4 * codeCnt);
|
||||||
|
msgh_size = (mach_msg_size_t)(sizeof(Request) - 904) + msgh_size_delta;
|
||||||
|
InP = (Request *) ((pointer_t) InP + msgh_size_delta - 8);
|
||||||
|
|
||||||
|
InP->flavor = *flavor;
|
||||||
|
|
||||||
|
if (old_stateCnt > 224) {
|
||||||
|
{ return MIG_ARRAY_TOO_LARGE; }
|
||||||
|
}
|
||||||
|
(void)memcpy((char *) InP->old_state, (const char *) old_state, 4 * old_stateCnt);
|
||||||
|
|
||||||
|
InP->old_stateCnt = old_stateCnt;
|
||||||
|
|
||||||
|
msgh_size += (4 * old_stateCnt);
|
||||||
|
InP = &Mess.In;
|
||||||
|
InP->Head.msgh_bits = MACH_MSGH_BITS_COMPLEX|
|
||||||
|
MACH_MSGH_BITS(19, MACH_MSG_TYPE_MAKE_SEND_ONCE);
|
||||||
|
/* msgh_size passed as argument */
|
||||||
|
InP->Head.msgh_request_port = exception_port;
|
||||||
|
InP->Head.msgh_reply_port = mig_get_reply_port();
|
||||||
|
InP->Head.msgh_id = 2403;
|
||||||
|
|
||||||
|
/* BEGIN VOUCHER CODE */
|
||||||
|
|
||||||
|
#ifdef USING_VOUCHERS
|
||||||
|
if (voucher_mach_msg_set != NULL) {
|
||||||
|
voucher_mach_msg_set(&InP->Head);
|
||||||
|
}
|
||||||
|
#endif // USING_VOUCHERS
|
||||||
|
|
||||||
|
/* END VOUCHER CODE */
|
||||||
|
|
||||||
|
__BeforeSendRpc(2403, "exception_raise_state_identity")
|
||||||
|
msg_result = mach_msg(&InP->Head, MACH_SEND_MSG|MACH_RCV_MSG|MACH_MSG_OPTION_NONE, msgh_size, (mach_msg_size_t)sizeof(Reply), InP->Head.msgh_reply_port, MACH_MSG_TIMEOUT_NONE, MACH_PORT_NULL);
|
||||||
|
__AfterSendRpc(2403, "exception_raise_state_identity")
|
||||||
|
if (msg_result != MACH_MSG_SUCCESS) {
|
||||||
|
__MachMsgErrorWithoutTimeout(msg_result);
|
||||||
|
{ return msg_result; }
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#if defined(__MIG_check__Reply__exception_raise_state_identity_t__defined)
|
||||||
|
check_result = __MIG_check__Reply__exception_raise_state_identity_t((__Reply__exception_raise_state_identity_t *)Out0P);
|
||||||
|
if (check_result != MACH_MSG_SUCCESS)
|
||||||
|
{ return check_result; }
|
||||||
|
#endif /* defined(__MIG_check__Reply__exception_raise_state_identity_t__defined) */
|
||||||
|
|
||||||
|
*flavor = Out0P->flavor;
|
||||||
|
|
||||||
|
if (Out0P->new_stateCnt > 224) {
|
||||||
|
(void)memcpy((char *) new_state, (const char *) Out0P->new_state, 4 * 224);
|
||||||
|
*new_stateCnt = Out0P->new_stateCnt;
|
||||||
|
{ return MIG_ARRAY_TOO_LARGE; }
|
||||||
|
}
|
||||||
|
(void)memcpy((char *) new_state, (const char *) Out0P->new_state, 4 * Out0P->new_stateCnt);
|
||||||
|
|
||||||
|
*new_stateCnt = Out0P->new_stateCnt;
|
||||||
|
|
||||||
|
return KERN_SUCCESS;
|
||||||
|
}
|
112
vendor/github.com/derekparker/delve/pkg/proc/exec_darwin.c
generated
vendored
Normal file
112
vendor/github.com/derekparker/delve/pkg/proc/exec_darwin.c
generated
vendored
Normal file
@ -0,0 +1,112 @@
|
|||||||
|
#include "exec_darwin.h"
|
||||||
|
#include "stdio.h"
|
||||||
|
|
||||||
|
extern char** environ;
|
||||||
|
|
||||||
|
int
|
||||||
|
close_exec_pipe(int fd[2]) {
|
||||||
|
if (pipe(fd) < 0) return -1;
|
||||||
|
if (fcntl(fd[0], F_SETFD, FD_CLOEXEC) < 0) return -1;
|
||||||
|
if (fcntl(fd[1], F_SETFD, FD_CLOEXEC) < 0) return -1;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
fork_exec(char *argv0, char **argv, int size,
|
||||||
|
char *wd,
|
||||||
|
task_t *task,
|
||||||
|
mach_port_t *port_set,
|
||||||
|
mach_port_t *exception_port,
|
||||||
|
mach_port_t *notification_port)
|
||||||
|
{
|
||||||
|
// Since we're using mach exceptions instead of signals,
|
||||||
|
// we need to coordinate between parent and child via pipes
|
||||||
|
// to ensure that the parent has set the exception ports on
|
||||||
|
// the child task before it execs.
|
||||||
|
int fd[2];
|
||||||
|
if (close_exec_pipe(fd) < 0) return -1;
|
||||||
|
|
||||||
|
// Create another pipe to signal the parent on exec.
|
||||||
|
int efd[2];
|
||||||
|
if (close_exec_pipe(efd) < 0) return -1;
|
||||||
|
|
||||||
|
kern_return_t kret;
|
||||||
|
pid_t pid = fork();
|
||||||
|
if (pid > 0) {
|
||||||
|
// In parent.
|
||||||
|
close(fd[0]);
|
||||||
|
close(efd[1]);
|
||||||
|
kret = acquire_mach_task(pid, task, port_set, exception_port, notification_port);
|
||||||
|
if (kret != KERN_SUCCESS) return -1;
|
||||||
|
|
||||||
|
char msg = 'c';
|
||||||
|
write(fd[1], &msg, 1);
|
||||||
|
close(fd[1]);
|
||||||
|
|
||||||
|
char w;
|
||||||
|
size_t n = read(efd[0], &w, 1);
|
||||||
|
close(efd[0]);
|
||||||
|
if (n != 0) {
|
||||||
|
// Child died, reap it.
|
||||||
|
waitpid(pid, NULL, 0);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return pid;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fork succeeded, we are in the child.
|
||||||
|
int pret, cret;
|
||||||
|
char sig;
|
||||||
|
|
||||||
|
close(fd[1]);
|
||||||
|
read(fd[0], &sig, 1);
|
||||||
|
close(fd[0]);
|
||||||
|
|
||||||
|
// Create a new process group.
|
||||||
|
if (setpgid(0, 0) < 0) {
|
||||||
|
perror("setpgid");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set errno to zero before a call to ptrace.
|
||||||
|
// It is documented that ptrace can return -1 even
|
||||||
|
// for successful calls.
|
||||||
|
errno = 0;
|
||||||
|
pret = ptrace(PT_TRACE_ME, 0, 0, 0);
|
||||||
|
if (pret != 0 && errno != 0) {
|
||||||
|
perror("ptrace");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Change working directory if wd is not empty.
|
||||||
|
if (wd && wd[0]) {
|
||||||
|
errno = 0;
|
||||||
|
cret = chdir(wd);
|
||||||
|
if (cret != 0 && errno != 0) {
|
||||||
|
char *error_msg;
|
||||||
|
asprintf(&error_msg, "%s '%s'", "chdir", wd);
|
||||||
|
perror(error_msg);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
pret = ptrace(PT_SIGEXC, 0, 0, 0);
|
||||||
|
if (pret != 0 && errno != 0) {
|
||||||
|
perror("ptrace");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
sleep(1);
|
||||||
|
|
||||||
|
// Create the child process.
|
||||||
|
execve(argv0, argv, environ);
|
||||||
|
|
||||||
|
// We should never reach here, but if we did something went wrong.
|
||||||
|
// Write a message to parent to alert that exec failed.
|
||||||
|
char msg = 'd';
|
||||||
|
write(efd[1], &msg, 1);
|
||||||
|
close(efd[1]);
|
||||||
|
|
||||||
|
exit(1);
|
||||||
|
}
|
10
vendor/github.com/derekparker/delve/pkg/proc/exec_darwin.h
generated
vendored
Normal file
10
vendor/github.com/derekparker/delve/pkg/proc/exec_darwin.h
generated
vendored
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
#include "proc_darwin.h"
|
||||||
|
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <sys/ptrace.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
|
||||||
|
int
|
||||||
|
fork_exec(char *, char **, int, char *, task_t*, mach_port_t*, mach_port_t*, mach_port_t*);
|
117
vendor/github.com/derekparker/delve/pkg/proc/go_version.go
generated
vendored
Normal file
117
vendor/github.com/derekparker/delve/pkg/proc/go_version.go
generated
vendored
Normal file
@ -0,0 +1,117 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// GoVersion represents the Go version of
|
||||||
|
// the Go compiler version used to compile
|
||||||
|
// the target binary.
|
||||||
|
type GoVersion struct {
|
||||||
|
Major int
|
||||||
|
Minor int
|
||||||
|
Rev int
|
||||||
|
Beta int
|
||||||
|
RC int
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
GoVer18Beta = GoVersion{1, 8, -1, 0, 0}
|
||||||
|
)
|
||||||
|
|
||||||
|
func ParseVersionString(ver string) (GoVersion, bool) {
|
||||||
|
var r GoVersion
|
||||||
|
var err1, err2, err3 error
|
||||||
|
|
||||||
|
if strings.HasPrefix(ver, "devel") {
|
||||||
|
return GoVersion{-1, 0, 0, 0, 0}, true
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.HasPrefix(ver, "go") {
|
||||||
|
ver := strings.Split(ver, " ")[0]
|
||||||
|
v := strings.SplitN(ver[2:], ".", 3)
|
||||||
|
switch len(v) {
|
||||||
|
case 2:
|
||||||
|
r.Major, err1 = strconv.Atoi(v[0])
|
||||||
|
vr := strings.SplitN(v[1], "beta", 2)
|
||||||
|
if len(vr) == 2 {
|
||||||
|
r.Beta, err3 = strconv.Atoi(vr[1])
|
||||||
|
} else {
|
||||||
|
vr = strings.SplitN(v[1], "rc", 2)
|
||||||
|
if len(vr) == 2 {
|
||||||
|
r.RC, err3 = strconv.Atoi(vr[1])
|
||||||
|
} else {
|
||||||
|
r.Minor, err2 = strconv.Atoi(v[1])
|
||||||
|
if err2 != nil {
|
||||||
|
return GoVersion{}, false
|
||||||
|
}
|
||||||
|
return r, true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
r.Minor, err2 = strconv.Atoi(vr[0])
|
||||||
|
r.Rev = -1
|
||||||
|
|
||||||
|
if err1 != nil || err2 != nil || err3 != nil {
|
||||||
|
return GoVersion{}, false
|
||||||
|
}
|
||||||
|
|
||||||
|
return r, true
|
||||||
|
|
||||||
|
case 3:
|
||||||
|
|
||||||
|
r.Major, err1 = strconv.Atoi(v[0])
|
||||||
|
r.Minor, err2 = strconv.Atoi(v[1])
|
||||||
|
r.Rev, err3 = strconv.Atoi(v[2])
|
||||||
|
if err1 != nil || err2 != nil || err3 != nil {
|
||||||
|
return GoVersion{}, false
|
||||||
|
}
|
||||||
|
|
||||||
|
return r, true
|
||||||
|
|
||||||
|
default:
|
||||||
|
return GoVersion{}, false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return GoVersion{}, false
|
||||||
|
}
|
||||||
|
|
||||||
|
// AfterOrEqual returns whether one GoVersion is after or
|
||||||
|
// equal to the other.
|
||||||
|
func (v *GoVersion) AfterOrEqual(b GoVersion) bool {
|
||||||
|
if v.Major < b.Major {
|
||||||
|
return false
|
||||||
|
} else if v.Major > b.Major {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
if v.Minor < b.Minor {
|
||||||
|
return false
|
||||||
|
} else if v.Minor > b.Minor {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
if v.Rev < b.Rev {
|
||||||
|
return false
|
||||||
|
} else if v.Rev > b.Rev {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
if v.Beta < b.Beta {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if v.RC < b.RC {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsDevel returns whether the GoVersion
|
||||||
|
// is a development version.
|
||||||
|
func (v *GoVersion) IsDevel() bool {
|
||||||
|
return v.Major < 0
|
||||||
|
}
|
119
vendor/github.com/derekparker/delve/pkg/proc/mach_exc.defs
generated
vendored
Normal file
119
vendor/github.com/derekparker/delve/pkg/proc/mach_exc.defs
generated
vendored
Normal file
@ -0,0 +1,119 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2006 Apple Computer, Inc. All rights reserved.
|
||||||
|
*
|
||||||
|
* @APPLE_OSREFERENCE_LICENSE_HEADER_START@
|
||||||
|
*
|
||||||
|
* This file contains Original Code and/or Modifications of Original Code
|
||||||
|
* as defined in and that are subject to the Apple Public Source License
|
||||||
|
* Version 2.0 (the 'License'). You may not use this file except in
|
||||||
|
* compliance with the License. The rights granted to you under the License
|
||||||
|
* may not be used to create, or enable the creation or redistribution of,
|
||||||
|
* unlawful or unlicensed copies of an Apple operating system, or to
|
||||||
|
* circumvent, violate, or enable the circumvention or violation of, any
|
||||||
|
* terms of an Apple operating system software license agreement.
|
||||||
|
*
|
||||||
|
* Please obtain a copy of the License at
|
||||||
|
* http://www.opensource.apple.com/apsl/ and read it before using this file.
|
||||||
|
*
|
||||||
|
* The Original Code and all software distributed under the License are
|
||||||
|
* distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
|
||||||
|
* EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
|
||||||
|
* INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
|
||||||
|
* Please see the License for the specific language governing rights and
|
||||||
|
* limitations under the License.
|
||||||
|
*
|
||||||
|
* @APPLE_OSREFERENCE_LICENSE_HEADER_END@
|
||||||
|
*/
|
||||||
|
/*
|
||||||
|
* @OSF_COPYRIGHT@
|
||||||
|
*/
|
||||||
|
/*
|
||||||
|
* Mach Operating System
|
||||||
|
* Copyright (c) 1991,1990,1989,1988,1987 Carnegie Mellon University
|
||||||
|
* All Rights Reserved.
|
||||||
|
*
|
||||||
|
* Permission to use, copy, modify and distribute this software and its
|
||||||
|
* documentation is hereby granted, provided that both the copyright
|
||||||
|
* notice and this permission notice appear in all copies of the
|
||||||
|
* software, derivative works or modified versions, and any portions
|
||||||
|
* thereof, and that both notices appear in supporting documentation.
|
||||||
|
*
|
||||||
|
* CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
|
||||||
|
* CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
|
||||||
|
* ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
|
||||||
|
*
|
||||||
|
* Carnegie Mellon requests users of this software to return to
|
||||||
|
*
|
||||||
|
* Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
|
||||||
|
* School of Computer Science
|
||||||
|
* Carnegie Mellon University
|
||||||
|
* Pittsburgh PA 15213-3890
|
||||||
|
*
|
||||||
|
* any improvements or extensions that they make and grant Carnegie Mellon
|
||||||
|
* the rights to redistribute these changes.
|
||||||
|
*/
|
||||||
|
/*
|
||||||
|
*/
|
||||||
|
/*
|
||||||
|
* Abstract:
|
||||||
|
* MiG definitions file for Mach exception interface.
|
||||||
|
*/
|
||||||
|
|
||||||
|
subsystem
|
||||||
|
#if KERNEL_USER
|
||||||
|
KernelUser
|
||||||
|
#endif
|
||||||
|
mach_exc 2405;
|
||||||
|
|
||||||
|
#include <mach/std_types.defs>
|
||||||
|
#include <mach/mach_types.defs>
|
||||||
|
|
||||||
|
ServerPrefix catch_;
|
||||||
|
|
||||||
|
type mach_exception_data_t = array[*:2] of int64_t;
|
||||||
|
type exception_type_t = int;
|
||||||
|
|
||||||
|
routine mach_exception_raise(
|
||||||
|
#if KERNEL_USER
|
||||||
|
exception_port : mach_port_move_send_t;
|
||||||
|
thread : mach_port_move_send_t;
|
||||||
|
task : mach_port_move_send_t;
|
||||||
|
#else /* KERNEL_USER */
|
||||||
|
exception_port : mach_port_t;
|
||||||
|
thread : mach_port_t;
|
||||||
|
task : mach_port_t;
|
||||||
|
#endif /* KERNEL_USER */
|
||||||
|
exception : exception_type_t;
|
||||||
|
code : mach_exception_data_t
|
||||||
|
);
|
||||||
|
|
||||||
|
routine mach_exception_raise_state(
|
||||||
|
#if KERNEL_USER
|
||||||
|
exception_port : mach_port_move_send_t;
|
||||||
|
#else /* KERNEL_USER */
|
||||||
|
exception_port : mach_port_t;
|
||||||
|
#endif /* KERNEL_USER */
|
||||||
|
exception : exception_type_t;
|
||||||
|
code : mach_exception_data_t, const;
|
||||||
|
inout flavor : int;
|
||||||
|
old_state : thread_state_t, const;
|
||||||
|
out new_state : thread_state_t);
|
||||||
|
|
||||||
|
routine mach_exception_raise_state_identity(
|
||||||
|
#if KERNEL_USER
|
||||||
|
exception_port : mach_port_move_send_t;
|
||||||
|
thread : mach_port_move_send_t;
|
||||||
|
task : mach_port_move_send_t;
|
||||||
|
#else /* KERNEL_USER */
|
||||||
|
exception_port : mach_port_t;
|
||||||
|
thread : mach_port_t;
|
||||||
|
task : mach_port_t;
|
||||||
|
#endif /* KERNEL_USER */
|
||||||
|
exception : exception_type_t;
|
||||||
|
code : mach_exception_data_t;
|
||||||
|
inout flavor : int;
|
||||||
|
old_state : thread_state_t;
|
||||||
|
out new_state : thread_state_t);
|
||||||
|
|
||||||
|
/* vim: set ft=c : */
|
283
vendor/github.com/derekparker/delve/pkg/proc/mach_exc.h
generated
vendored
Normal file
283
vendor/github.com/derekparker/delve/pkg/proc/mach_exc.h
generated
vendored
Normal file
@ -0,0 +1,283 @@
|
|||||||
|
#ifndef _mach_exc_user_
|
||||||
|
#define _mach_exc_user_
|
||||||
|
|
||||||
|
/* Module mach_exc */
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
#include <mach/ndr.h>
|
||||||
|
#include <mach/boolean.h>
|
||||||
|
#include <mach/kern_return.h>
|
||||||
|
#include <mach/notify.h>
|
||||||
|
#include <mach/mach_types.h>
|
||||||
|
#include <mach/message.h>
|
||||||
|
#include <mach/mig_errors.h>
|
||||||
|
#include <mach/port.h>
|
||||||
|
|
||||||
|
/* BEGIN VOUCHER CODE */
|
||||||
|
|
||||||
|
#ifndef KERNEL
|
||||||
|
#if defined(__has_include)
|
||||||
|
#if __has_include(<mach/mig_voucher_support.h>)
|
||||||
|
#ifndef USING_VOUCHERS
|
||||||
|
#define USING_VOUCHERS
|
||||||
|
#endif
|
||||||
|
#ifndef __VOUCHER_FORWARD_TYPE_DECLS__
|
||||||
|
#define __VOUCHER_FORWARD_TYPE_DECLS__
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
extern boolean_t voucher_mach_msg_set(mach_msg_header_t *msg) __attribute__((weak_import));
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#endif // __VOUCHER_FORWARD_TYPE_DECLS__
|
||||||
|
#endif // __has_include(<mach/mach_voucher_types.h>)
|
||||||
|
#endif // __has_include
|
||||||
|
#endif // !KERNEL
|
||||||
|
|
||||||
|
/* END VOUCHER CODE */
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef AUTOTEST
|
||||||
|
#ifndef FUNCTION_PTR_T
|
||||||
|
#define FUNCTION_PTR_T
|
||||||
|
typedef void (*function_ptr_t)(mach_port_t, char *, mach_msg_type_number_t);
|
||||||
|
typedef struct {
|
||||||
|
char *name;
|
||||||
|
function_ptr_t function;
|
||||||
|
} function_table_entry;
|
||||||
|
typedef function_table_entry *function_table_t;
|
||||||
|
#endif /* FUNCTION_PTR_T */
|
||||||
|
#endif /* AUTOTEST */
|
||||||
|
|
||||||
|
#ifndef mach_exc_MSG_COUNT
|
||||||
|
#define mach_exc_MSG_COUNT 3
|
||||||
|
#endif /* mach_exc_MSG_COUNT */
|
||||||
|
|
||||||
|
#include <mach/std_types.h>
|
||||||
|
#include <mach/mig.h>
|
||||||
|
#include <mach/mig.h>
|
||||||
|
#include <mach/mach_types.h>
|
||||||
|
|
||||||
|
#ifdef __BeforeMigUserHeader
|
||||||
|
__BeforeMigUserHeader
|
||||||
|
#endif /* __BeforeMigUserHeader */
|
||||||
|
|
||||||
|
#include <sys/cdefs.h>
|
||||||
|
__BEGIN_DECLS
|
||||||
|
|
||||||
|
|
||||||
|
/* Routine mach_exception_raise */
|
||||||
|
#ifdef mig_external
|
||||||
|
mig_external
|
||||||
|
#else
|
||||||
|
extern
|
||||||
|
#endif /* mig_external */
|
||||||
|
kern_return_t mach_exception_raise
|
||||||
|
(
|
||||||
|
mach_port_t exception_port,
|
||||||
|
mach_port_t thread,
|
||||||
|
mach_port_t task,
|
||||||
|
exception_type_t exception,
|
||||||
|
mach_exception_data_t code,
|
||||||
|
mach_msg_type_number_t codeCnt
|
||||||
|
);
|
||||||
|
|
||||||
|
/* Routine mach_exception_raise_state */
|
||||||
|
#ifdef mig_external
|
||||||
|
mig_external
|
||||||
|
#else
|
||||||
|
extern
|
||||||
|
#endif /* mig_external */
|
||||||
|
kern_return_t mach_exception_raise_state
|
||||||
|
(
|
||||||
|
mach_port_t exception_port,
|
||||||
|
exception_type_t exception,
|
||||||
|
const mach_exception_data_t code,
|
||||||
|
mach_msg_type_number_t codeCnt,
|
||||||
|
int *flavor,
|
||||||
|
const thread_state_t old_state,
|
||||||
|
mach_msg_type_number_t old_stateCnt,
|
||||||
|
thread_state_t new_state,
|
||||||
|
mach_msg_type_number_t *new_stateCnt
|
||||||
|
);
|
||||||
|
|
||||||
|
/* Routine mach_exception_raise_state_identity */
|
||||||
|
#ifdef mig_external
|
||||||
|
mig_external
|
||||||
|
#else
|
||||||
|
extern
|
||||||
|
#endif /* mig_external */
|
||||||
|
kern_return_t mach_exception_raise_state_identity
|
||||||
|
(
|
||||||
|
mach_port_t exception_port,
|
||||||
|
mach_port_t thread,
|
||||||
|
mach_port_t task,
|
||||||
|
exception_type_t exception,
|
||||||
|
mach_exception_data_t code,
|
||||||
|
mach_msg_type_number_t codeCnt,
|
||||||
|
int *flavor,
|
||||||
|
thread_state_t old_state,
|
||||||
|
mach_msg_type_number_t old_stateCnt,
|
||||||
|
thread_state_t new_state,
|
||||||
|
mach_msg_type_number_t *new_stateCnt
|
||||||
|
);
|
||||||
|
|
||||||
|
__END_DECLS
|
||||||
|
|
||||||
|
/********************** Caution **************************/
|
||||||
|
/* The following data types should be used to calculate */
|
||||||
|
/* maximum message sizes only. The actual message may be */
|
||||||
|
/* smaller, and the position of the arguments within the */
|
||||||
|
/* message layout may vary from what is presented here. */
|
||||||
|
/* For example, if any of the arguments are variable- */
|
||||||
|
/* sized, and less than the maximum is sent, the data */
|
||||||
|
/* will be packed tight in the actual message to reduce */
|
||||||
|
/* the presence of holes. */
|
||||||
|
/********************** Caution **************************/
|
||||||
|
|
||||||
|
/* typedefs for all requests */
|
||||||
|
|
||||||
|
#ifndef __Request__mach_exc_subsystem__defined
|
||||||
|
#define __Request__mach_exc_subsystem__defined
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
/* start of the kernel processed data */
|
||||||
|
mach_msg_body_t msgh_body;
|
||||||
|
mach_msg_port_descriptor_t thread;
|
||||||
|
mach_msg_port_descriptor_t task;
|
||||||
|
/* end of the kernel processed data */
|
||||||
|
NDR_record_t NDR;
|
||||||
|
exception_type_t exception;
|
||||||
|
mach_msg_type_number_t codeCnt;
|
||||||
|
int64_t code[2];
|
||||||
|
} __Request__mach_exception_raise_t;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
exception_type_t exception;
|
||||||
|
mach_msg_type_number_t codeCnt;
|
||||||
|
int64_t code[2];
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t old_stateCnt;
|
||||||
|
natural_t old_state[224];
|
||||||
|
} __Request__mach_exception_raise_state_t;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
/* start of the kernel processed data */
|
||||||
|
mach_msg_body_t msgh_body;
|
||||||
|
mach_msg_port_descriptor_t thread;
|
||||||
|
mach_msg_port_descriptor_t task;
|
||||||
|
/* end of the kernel processed data */
|
||||||
|
NDR_record_t NDR;
|
||||||
|
exception_type_t exception;
|
||||||
|
mach_msg_type_number_t codeCnt;
|
||||||
|
int64_t code[2];
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t old_stateCnt;
|
||||||
|
natural_t old_state[224];
|
||||||
|
} __Request__mach_exception_raise_state_identity_t;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
#endif /* !__Request__mach_exc_subsystem__defined */
|
||||||
|
|
||||||
|
/* union of all requests */
|
||||||
|
|
||||||
|
#ifndef __RequestUnion__mach_exc_subsystem__defined
|
||||||
|
#define __RequestUnion__mach_exc_subsystem__defined
|
||||||
|
union __RequestUnion__mach_exc_subsystem {
|
||||||
|
__Request__mach_exception_raise_t Request_mach_exception_raise;
|
||||||
|
__Request__mach_exception_raise_state_t Request_mach_exception_raise_state;
|
||||||
|
__Request__mach_exception_raise_state_identity_t Request_mach_exception_raise_state_identity;
|
||||||
|
};
|
||||||
|
#endif /* !__RequestUnion__mach_exc_subsystem__defined */
|
||||||
|
/* typedefs for all replies */
|
||||||
|
|
||||||
|
#ifndef __Reply__mach_exc_subsystem__defined
|
||||||
|
#define __Reply__mach_exc_subsystem__defined
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
} __Reply__mach_exception_raise_t;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t new_stateCnt;
|
||||||
|
natural_t new_state[224];
|
||||||
|
} __Reply__mach_exception_raise_state_t;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t new_stateCnt;
|
||||||
|
natural_t new_state[224];
|
||||||
|
} __Reply__mach_exception_raise_state_identity_t;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
#endif /* !__Reply__mach_exc_subsystem__defined */
|
||||||
|
|
||||||
|
/* union of all replies */
|
||||||
|
|
||||||
|
#ifndef __ReplyUnion__mach_exc_subsystem__defined
|
||||||
|
#define __ReplyUnion__mach_exc_subsystem__defined
|
||||||
|
union __ReplyUnion__mach_exc_subsystem {
|
||||||
|
__Reply__mach_exception_raise_t Reply_mach_exception_raise;
|
||||||
|
__Reply__mach_exception_raise_state_t Reply_mach_exception_raise_state;
|
||||||
|
__Reply__mach_exception_raise_state_identity_t Reply_mach_exception_raise_state_identity;
|
||||||
|
};
|
||||||
|
#endif /* !__RequestUnion__mach_exc_subsystem__defined */
|
||||||
|
|
||||||
|
#ifndef subsystem_to_name_map_mach_exc
|
||||||
|
#define subsystem_to_name_map_mach_exc \
|
||||||
|
{ "mach_exception_raise", 2401 },\
|
||||||
|
{ "mach_exception_raise_state", 2402 },\
|
||||||
|
{ "mach_exception_raise_state_identity", 2403 }
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __AfterMigUserHeader
|
||||||
|
__AfterMigUserHeader
|
||||||
|
#endif /* __AfterMigUserHeader */
|
||||||
|
|
||||||
|
#endif /* _mach_exc_user_ */
|
768
vendor/github.com/derekparker/delve/pkg/proc/mach_exc_user_darwin.c
generated
vendored
Normal file
768
vendor/github.com/derekparker/delve/pkg/proc/mach_exc_user_darwin.c
generated
vendored
Normal file
@ -0,0 +1,768 @@
|
|||||||
|
/*
|
||||||
|
* IDENTIFICATION:
|
||||||
|
* stub generated Sat Feb 21 18:10:52 2015
|
||||||
|
* with a MiG generated by bootstrap_cmds-91
|
||||||
|
* OPTIONS:
|
||||||
|
*/
|
||||||
|
#define __MIG_check__Reply__mach_exc_subsystem__ 1
|
||||||
|
|
||||||
|
#include "mach_exc.h"
|
||||||
|
|
||||||
|
|
||||||
|
#ifndef mig_internal
|
||||||
|
#define mig_internal static __inline__
|
||||||
|
#endif /* mig_internal */
|
||||||
|
|
||||||
|
#ifndef mig_external
|
||||||
|
#define mig_external
|
||||||
|
#endif /* mig_external */
|
||||||
|
|
||||||
|
#if !defined(__MigTypeCheck) && defined(TypeCheck)
|
||||||
|
#define __MigTypeCheck TypeCheck /* Legacy setting */
|
||||||
|
#endif /* !defined(__MigTypeCheck) */
|
||||||
|
|
||||||
|
#if !defined(__MigKernelSpecificCode) && defined(_MIG_KERNEL_SPECIFIC_CODE_)
|
||||||
|
#define __MigKernelSpecificCode _MIG_KERNEL_SPECIFIC_CODE_ /* Legacy setting */
|
||||||
|
#endif /* !defined(__MigKernelSpecificCode) */
|
||||||
|
|
||||||
|
#ifndef LimitCheck
|
||||||
|
#define LimitCheck 0
|
||||||
|
#endif /* LimitCheck */
|
||||||
|
|
||||||
|
#ifndef min
|
||||||
|
#define min(a,b) ( ((a) < (b))? (a): (b) )
|
||||||
|
#endif /* min */
|
||||||
|
|
||||||
|
#if !defined(_WALIGN_)
|
||||||
|
#define _WALIGN_(x) (((x) + 3) & ~3)
|
||||||
|
#endif /* !defined(_WALIGN_) */
|
||||||
|
|
||||||
|
#if !defined(_WALIGNSZ_)
|
||||||
|
#define _WALIGNSZ_(x) _WALIGN_(sizeof(x))
|
||||||
|
#endif /* !defined(_WALIGNSZ_) */
|
||||||
|
|
||||||
|
#ifndef UseStaticTemplates
|
||||||
|
#define UseStaticTemplates 0
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
#ifndef __MachMsgErrorWithTimeout
|
||||||
|
#define __MachMsgErrorWithTimeout(_R_) { \
|
||||||
|
switch (_R_) { \
|
||||||
|
case MACH_SEND_INVALID_DATA: \
|
||||||
|
case MACH_SEND_INVALID_DEST: \
|
||||||
|
case MACH_SEND_INVALID_HEADER: \
|
||||||
|
mig_put_reply_port(InP->Head.msgh_reply_port); \
|
||||||
|
break; \
|
||||||
|
case MACH_SEND_TIMED_OUT: \
|
||||||
|
case MACH_RCV_TIMED_OUT: \
|
||||||
|
default: \
|
||||||
|
mig_dealloc_reply_port(InP->Head.msgh_reply_port); \
|
||||||
|
} \
|
||||||
|
}
|
||||||
|
#endif /* __MachMsgErrorWithTimeout */
|
||||||
|
|
||||||
|
#ifndef __MachMsgErrorWithoutTimeout
|
||||||
|
#define __MachMsgErrorWithoutTimeout(_R_) { \
|
||||||
|
switch (_R_) { \
|
||||||
|
case MACH_SEND_INVALID_DATA: \
|
||||||
|
case MACH_SEND_INVALID_DEST: \
|
||||||
|
case MACH_SEND_INVALID_HEADER: \
|
||||||
|
mig_put_reply_port(InP->Head.msgh_reply_port); \
|
||||||
|
break; \
|
||||||
|
default: \
|
||||||
|
mig_dealloc_reply_port(InP->Head.msgh_reply_port); \
|
||||||
|
} \
|
||||||
|
}
|
||||||
|
#endif /* __MachMsgErrorWithoutTimeout */
|
||||||
|
|
||||||
|
#ifndef __DeclareSendRpc
|
||||||
|
#define __DeclareSendRpc(_NUM_, _NAME_)
|
||||||
|
#endif /* __DeclareSendRpc */
|
||||||
|
|
||||||
|
#ifndef __BeforeSendRpc
|
||||||
|
#define __BeforeSendRpc(_NUM_, _NAME_)
|
||||||
|
#endif /* __BeforeSendRpc */
|
||||||
|
|
||||||
|
#ifndef __AfterSendRpc
|
||||||
|
#define __AfterSendRpc(_NUM_, _NAME_)
|
||||||
|
#endif /* __AfterSendRpc */
|
||||||
|
|
||||||
|
#ifndef __DeclareSendSimple
|
||||||
|
#define __DeclareSendSimple(_NUM_, _NAME_)
|
||||||
|
#endif /* __DeclareSendSimple */
|
||||||
|
|
||||||
|
#ifndef __BeforeSendSimple
|
||||||
|
#define __BeforeSendSimple(_NUM_, _NAME_)
|
||||||
|
#endif /* __BeforeSendSimple */
|
||||||
|
|
||||||
|
#ifndef __AfterSendSimple
|
||||||
|
#define __AfterSendSimple(_NUM_, _NAME_)
|
||||||
|
#endif /* __AfterSendSimple */
|
||||||
|
|
||||||
|
#define msgh_request_port msgh_remote_port
|
||||||
|
#define msgh_reply_port msgh_local_port
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#if ( __MigTypeCheck )
|
||||||
|
#if __MIG_check__Reply__mach_exc_subsystem__
|
||||||
|
#if !defined(__MIG_check__Reply__mach_exception_raise_t__defined)
|
||||||
|
#define __MIG_check__Reply__mach_exception_raise_t__defined
|
||||||
|
|
||||||
|
mig_internal kern_return_t __MIG_check__Reply__mach_exception_raise_t(__Reply__mach_exception_raise_t *Out0P)
|
||||||
|
{
|
||||||
|
|
||||||
|
typedef __Reply__mach_exception_raise_t __Reply;
|
||||||
|
if (Out0P->Head.msgh_id != 2505) {
|
||||||
|
if (Out0P->Head.msgh_id == MACH_NOTIFY_SEND_ONCE)
|
||||||
|
{ return MIG_SERVER_DIED; }
|
||||||
|
else
|
||||||
|
{ return MIG_REPLY_MISMATCH; }
|
||||||
|
}
|
||||||
|
|
||||||
|
#if __MigTypeCheck
|
||||||
|
if ((Out0P->Head.msgh_bits & MACH_MSGH_BITS_COMPLEX) ||
|
||||||
|
(Out0P->Head.msgh_size != (mach_msg_size_t)sizeof(__Reply)))
|
||||||
|
{ return MIG_TYPE_ERROR ; }
|
||||||
|
#endif /* __MigTypeCheck */
|
||||||
|
|
||||||
|
{
|
||||||
|
return Out0P->RetCode;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif /* !defined(__MIG_check__Reply__mach_exception_raise_t__defined) */
|
||||||
|
#endif /* __MIG_check__Reply__mach_exc_subsystem__ */
|
||||||
|
#endif /* ( __MigTypeCheck ) */
|
||||||
|
|
||||||
|
|
||||||
|
/* Routine mach_exception_raise */
|
||||||
|
mig_external kern_return_t mach_exception_raise
|
||||||
|
(
|
||||||
|
mach_port_t exception_port,
|
||||||
|
mach_port_t thread,
|
||||||
|
mach_port_t task,
|
||||||
|
exception_type_t exception,
|
||||||
|
mach_exception_data_t code,
|
||||||
|
mach_msg_type_number_t codeCnt
|
||||||
|
)
|
||||||
|
{
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
/* start of the kernel processed data */
|
||||||
|
mach_msg_body_t msgh_body;
|
||||||
|
mach_msg_port_descriptor_t thread;
|
||||||
|
mach_msg_port_descriptor_t task;
|
||||||
|
/* end of the kernel processed data */
|
||||||
|
NDR_record_t NDR;
|
||||||
|
exception_type_t exception;
|
||||||
|
mach_msg_type_number_t codeCnt;
|
||||||
|
int64_t code[2];
|
||||||
|
} Request;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
mach_msg_trailer_t trailer;
|
||||||
|
} Reply;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
} __Reply;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
/*
|
||||||
|
* typedef struct {
|
||||||
|
* mach_msg_header_t Head;
|
||||||
|
* NDR_record_t NDR;
|
||||||
|
* kern_return_t RetCode;
|
||||||
|
* } mig_reply_error_t;
|
||||||
|
*/
|
||||||
|
|
||||||
|
union {
|
||||||
|
Request In;
|
||||||
|
Reply Out;
|
||||||
|
} Mess;
|
||||||
|
|
||||||
|
Request *InP = &Mess.In;
|
||||||
|
Reply *Out0P = &Mess.Out;
|
||||||
|
|
||||||
|
mach_msg_return_t msg_result;
|
||||||
|
unsigned int msgh_size;
|
||||||
|
|
||||||
|
#ifdef __MIG_check__Reply__mach_exception_raise_t__defined
|
||||||
|
kern_return_t check_result;
|
||||||
|
#endif /* __MIG_check__Reply__mach_exception_raise_t__defined */
|
||||||
|
|
||||||
|
__DeclareSendRpc(2405, "mach_exception_raise")
|
||||||
|
|
||||||
|
#if UseStaticTemplates
|
||||||
|
const static mach_msg_port_descriptor_t threadTemplate = {
|
||||||
|
/* name = */ MACH_PORT_NULL,
|
||||||
|
/* pad1 = */ 0,
|
||||||
|
/* pad2 = */ 0,
|
||||||
|
/* disp = */ 19,
|
||||||
|
/* type = */ MACH_MSG_PORT_DESCRIPTOR,
|
||||||
|
};
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
#if UseStaticTemplates
|
||||||
|
const static mach_msg_port_descriptor_t taskTemplate = {
|
||||||
|
/* name = */ MACH_PORT_NULL,
|
||||||
|
/* pad1 = */ 0,
|
||||||
|
/* pad2 = */ 0,
|
||||||
|
/* disp = */ 19,
|
||||||
|
/* type = */ MACH_MSG_PORT_DESCRIPTOR,
|
||||||
|
};
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
InP->msgh_body.msgh_descriptor_count = 2;
|
||||||
|
#if UseStaticTemplates
|
||||||
|
InP->thread = threadTemplate;
|
||||||
|
InP->thread.name = thread;
|
||||||
|
#else /* UseStaticTemplates */
|
||||||
|
InP->thread.name = thread;
|
||||||
|
InP->thread.disposition = 19;
|
||||||
|
InP->thread.type = MACH_MSG_PORT_DESCRIPTOR;
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
#if UseStaticTemplates
|
||||||
|
InP->task = taskTemplate;
|
||||||
|
InP->task.name = task;
|
||||||
|
#else /* UseStaticTemplates */
|
||||||
|
InP->task.name = task;
|
||||||
|
InP->task.disposition = 19;
|
||||||
|
InP->task.type = MACH_MSG_PORT_DESCRIPTOR;
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
InP->NDR = NDR_record;
|
||||||
|
|
||||||
|
InP->exception = exception;
|
||||||
|
|
||||||
|
if (codeCnt > 2) {
|
||||||
|
{ return MIG_ARRAY_TOO_LARGE; }
|
||||||
|
}
|
||||||
|
(void)memcpy((char *) InP->code, (const char *) code, 8 * codeCnt);
|
||||||
|
|
||||||
|
InP->codeCnt = codeCnt;
|
||||||
|
|
||||||
|
msgh_size = (mach_msg_size_t)(sizeof(Request) - 16) + ((8 * codeCnt));
|
||||||
|
InP->Head.msgh_bits = MACH_MSGH_BITS_COMPLEX|
|
||||||
|
MACH_MSGH_BITS(19, MACH_MSG_TYPE_MAKE_SEND_ONCE);
|
||||||
|
/* msgh_size passed as argument */
|
||||||
|
InP->Head.msgh_request_port = exception_port;
|
||||||
|
InP->Head.msgh_reply_port = mig_get_reply_port();
|
||||||
|
InP->Head.msgh_id = 2405;
|
||||||
|
|
||||||
|
/* BEGIN VOUCHER CODE */
|
||||||
|
|
||||||
|
#ifdef USING_VOUCHERS
|
||||||
|
if (voucher_mach_msg_set != NULL) {
|
||||||
|
voucher_mach_msg_set(&InP->Head);
|
||||||
|
}
|
||||||
|
#endif // USING_VOUCHERS
|
||||||
|
|
||||||
|
/* END VOUCHER CODE */
|
||||||
|
|
||||||
|
__BeforeSendRpc(2405, "mach_exception_raise")
|
||||||
|
msg_result = mach_msg(&InP->Head, MACH_SEND_MSG|MACH_RCV_MSG|MACH_MSG_OPTION_NONE, msgh_size, (mach_msg_size_t)sizeof(Reply), InP->Head.msgh_reply_port, MACH_MSG_TIMEOUT_NONE, MACH_PORT_NULL);
|
||||||
|
__AfterSendRpc(2405, "mach_exception_raise")
|
||||||
|
if (msg_result != MACH_MSG_SUCCESS) {
|
||||||
|
__MachMsgErrorWithoutTimeout(msg_result);
|
||||||
|
{ return msg_result; }
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#if defined(__MIG_check__Reply__mach_exception_raise_t__defined)
|
||||||
|
check_result = __MIG_check__Reply__mach_exception_raise_t((__Reply__mach_exception_raise_t *)Out0P);
|
||||||
|
if (check_result != MACH_MSG_SUCCESS)
|
||||||
|
{ return check_result; }
|
||||||
|
#endif /* defined(__MIG_check__Reply__mach_exception_raise_t__defined) */
|
||||||
|
|
||||||
|
return KERN_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if ( __MigTypeCheck )
|
||||||
|
#if __MIG_check__Reply__mach_exc_subsystem__
|
||||||
|
#if !defined(__MIG_check__Reply__mach_exception_raise_state_t__defined)
|
||||||
|
#define __MIG_check__Reply__mach_exception_raise_state_t__defined
|
||||||
|
|
||||||
|
mig_internal kern_return_t __MIG_check__Reply__mach_exception_raise_state_t(__Reply__mach_exception_raise_state_t *Out0P)
|
||||||
|
{
|
||||||
|
|
||||||
|
typedef __Reply__mach_exception_raise_state_t __Reply;
|
||||||
|
#if __MigTypeCheck
|
||||||
|
unsigned int msgh_size;
|
||||||
|
#endif /* __MigTypeCheck */
|
||||||
|
|
||||||
|
if (Out0P->Head.msgh_id != 2506) {
|
||||||
|
if (Out0P->Head.msgh_id == MACH_NOTIFY_SEND_ONCE)
|
||||||
|
{ return MIG_SERVER_DIED; }
|
||||||
|
else
|
||||||
|
{ return MIG_REPLY_MISMATCH; }
|
||||||
|
}
|
||||||
|
|
||||||
|
#if __MigTypeCheck
|
||||||
|
msgh_size = Out0P->Head.msgh_size;
|
||||||
|
|
||||||
|
if ((Out0P->Head.msgh_bits & MACH_MSGH_BITS_COMPLEX) ||
|
||||||
|
((msgh_size > (mach_msg_size_t)sizeof(__Reply) || msgh_size < (mach_msg_size_t)(sizeof(__Reply) - 896)) &&
|
||||||
|
(msgh_size != (mach_msg_size_t)sizeof(mig_reply_error_t) ||
|
||||||
|
Out0P->RetCode == KERN_SUCCESS)))
|
||||||
|
{ return MIG_TYPE_ERROR ; }
|
||||||
|
#endif /* __MigTypeCheck */
|
||||||
|
|
||||||
|
if (Out0P->RetCode != KERN_SUCCESS) {
|
||||||
|
return ((mig_reply_error_t *)Out0P)->RetCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if __MigTypeCheck
|
||||||
|
if ( Out0P->new_stateCnt > 224 )
|
||||||
|
return MIG_TYPE_ERROR;
|
||||||
|
if (((msgh_size - (mach_msg_size_t)(sizeof(__Reply) - 896)) / 4< Out0P->new_stateCnt) ||
|
||||||
|
(msgh_size != (mach_msg_size_t)(sizeof(__Reply) - 896) + Out0P->new_stateCnt * 4))
|
||||||
|
{ return MIG_TYPE_ERROR ; }
|
||||||
|
#endif /* __MigTypeCheck */
|
||||||
|
|
||||||
|
return MACH_MSG_SUCCESS;
|
||||||
|
}
|
||||||
|
#endif /* !defined(__MIG_check__Reply__mach_exception_raise_state_t__defined) */
|
||||||
|
#endif /* __MIG_check__Reply__mach_exc_subsystem__ */
|
||||||
|
#endif /* ( __MigTypeCheck ) */
|
||||||
|
|
||||||
|
|
||||||
|
/* Routine mach_exception_raise_state */
|
||||||
|
mig_external kern_return_t mach_exception_raise_state
|
||||||
|
(
|
||||||
|
mach_port_t exception_port,
|
||||||
|
exception_type_t exception,
|
||||||
|
const mach_exception_data_t code,
|
||||||
|
mach_msg_type_number_t codeCnt,
|
||||||
|
int *flavor,
|
||||||
|
const thread_state_t old_state,
|
||||||
|
mach_msg_type_number_t old_stateCnt,
|
||||||
|
thread_state_t new_state,
|
||||||
|
mach_msg_type_number_t *new_stateCnt
|
||||||
|
)
|
||||||
|
{
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
exception_type_t exception;
|
||||||
|
mach_msg_type_number_t codeCnt;
|
||||||
|
int64_t code[2];
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t old_stateCnt;
|
||||||
|
natural_t old_state[224];
|
||||||
|
} Request;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t new_stateCnt;
|
||||||
|
natural_t new_state[224];
|
||||||
|
mach_msg_trailer_t trailer;
|
||||||
|
} Reply;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t new_stateCnt;
|
||||||
|
natural_t new_state[224];
|
||||||
|
} __Reply;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
/*
|
||||||
|
* typedef struct {
|
||||||
|
* mach_msg_header_t Head;
|
||||||
|
* NDR_record_t NDR;
|
||||||
|
* kern_return_t RetCode;
|
||||||
|
* } mig_reply_error_t;
|
||||||
|
*/
|
||||||
|
|
||||||
|
union {
|
||||||
|
Request In;
|
||||||
|
Reply Out;
|
||||||
|
} Mess;
|
||||||
|
|
||||||
|
Request *InP = &Mess.In;
|
||||||
|
Reply *Out0P = &Mess.Out;
|
||||||
|
|
||||||
|
mach_msg_return_t msg_result;
|
||||||
|
unsigned int msgh_size;
|
||||||
|
unsigned int msgh_size_delta;
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef __MIG_check__Reply__mach_exception_raise_state_t__defined
|
||||||
|
kern_return_t check_result;
|
||||||
|
#endif /* __MIG_check__Reply__mach_exception_raise_state_t__defined */
|
||||||
|
|
||||||
|
__DeclareSendRpc(2406, "mach_exception_raise_state")
|
||||||
|
|
||||||
|
InP->NDR = NDR_record;
|
||||||
|
|
||||||
|
InP->exception = exception;
|
||||||
|
|
||||||
|
if (codeCnt > 2) {
|
||||||
|
{ return MIG_ARRAY_TOO_LARGE; }
|
||||||
|
}
|
||||||
|
(void)memcpy((char *) InP->code, (const char *) code, 8 * codeCnt);
|
||||||
|
|
||||||
|
InP->codeCnt = codeCnt;
|
||||||
|
|
||||||
|
msgh_size_delta = (8 * codeCnt);
|
||||||
|
msgh_size = (mach_msg_size_t)(sizeof(Request) - 912) + msgh_size_delta;
|
||||||
|
InP = (Request *) ((pointer_t) InP + msgh_size_delta - 16);
|
||||||
|
|
||||||
|
InP->flavor = *flavor;
|
||||||
|
|
||||||
|
if (old_stateCnt > 224) {
|
||||||
|
{ return MIG_ARRAY_TOO_LARGE; }
|
||||||
|
}
|
||||||
|
(void)memcpy((char *) InP->old_state, (const char *) old_state, 4 * old_stateCnt);
|
||||||
|
|
||||||
|
InP->old_stateCnt = old_stateCnt;
|
||||||
|
|
||||||
|
msgh_size += (4 * old_stateCnt);
|
||||||
|
InP = &Mess.In;
|
||||||
|
InP->Head.msgh_bits =
|
||||||
|
MACH_MSGH_BITS(19, MACH_MSG_TYPE_MAKE_SEND_ONCE);
|
||||||
|
/* msgh_size passed as argument */
|
||||||
|
InP->Head.msgh_request_port = exception_port;
|
||||||
|
InP->Head.msgh_reply_port = mig_get_reply_port();
|
||||||
|
InP->Head.msgh_id = 2406;
|
||||||
|
|
||||||
|
/* BEGIN VOUCHER CODE */
|
||||||
|
|
||||||
|
#ifdef USING_VOUCHERS
|
||||||
|
if (voucher_mach_msg_set != NULL) {
|
||||||
|
voucher_mach_msg_set(&InP->Head);
|
||||||
|
}
|
||||||
|
#endif // USING_VOUCHERS
|
||||||
|
|
||||||
|
/* END VOUCHER CODE */
|
||||||
|
|
||||||
|
__BeforeSendRpc(2406, "mach_exception_raise_state")
|
||||||
|
msg_result = mach_msg(&InP->Head, MACH_SEND_MSG|MACH_RCV_MSG|MACH_MSG_OPTION_NONE, msgh_size, (mach_msg_size_t)sizeof(Reply), InP->Head.msgh_reply_port, MACH_MSG_TIMEOUT_NONE, MACH_PORT_NULL);
|
||||||
|
__AfterSendRpc(2406, "mach_exception_raise_state")
|
||||||
|
if (msg_result != MACH_MSG_SUCCESS) {
|
||||||
|
__MachMsgErrorWithoutTimeout(msg_result);
|
||||||
|
{ return msg_result; }
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#if defined(__MIG_check__Reply__mach_exception_raise_state_t__defined)
|
||||||
|
check_result = __MIG_check__Reply__mach_exception_raise_state_t((__Reply__mach_exception_raise_state_t *)Out0P);
|
||||||
|
if (check_result != MACH_MSG_SUCCESS)
|
||||||
|
{ return check_result; }
|
||||||
|
#endif /* defined(__MIG_check__Reply__mach_exception_raise_state_t__defined) */
|
||||||
|
|
||||||
|
*flavor = Out0P->flavor;
|
||||||
|
|
||||||
|
if (Out0P->new_stateCnt > 224) {
|
||||||
|
(void)memcpy((char *) new_state, (const char *) Out0P->new_state, 4 * 224);
|
||||||
|
*new_stateCnt = Out0P->new_stateCnt;
|
||||||
|
{ return MIG_ARRAY_TOO_LARGE; }
|
||||||
|
}
|
||||||
|
(void)memcpy((char *) new_state, (const char *) Out0P->new_state, 4 * Out0P->new_stateCnt);
|
||||||
|
|
||||||
|
*new_stateCnt = Out0P->new_stateCnt;
|
||||||
|
|
||||||
|
return KERN_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if ( __MigTypeCheck )
|
||||||
|
#if __MIG_check__Reply__mach_exc_subsystem__
|
||||||
|
#if !defined(__MIG_check__Reply__mach_exception_raise_state_identity_t__defined)
|
||||||
|
#define __MIG_check__Reply__mach_exception_raise_state_identity_t__defined
|
||||||
|
|
||||||
|
mig_internal kern_return_t __MIG_check__Reply__mach_exception_raise_state_identity_t(__Reply__mach_exception_raise_state_identity_t *Out0P)
|
||||||
|
{
|
||||||
|
|
||||||
|
typedef __Reply__mach_exception_raise_state_identity_t __Reply;
|
||||||
|
#if __MigTypeCheck
|
||||||
|
unsigned int msgh_size;
|
||||||
|
#endif /* __MigTypeCheck */
|
||||||
|
|
||||||
|
if (Out0P->Head.msgh_id != 2507) {
|
||||||
|
if (Out0P->Head.msgh_id == MACH_NOTIFY_SEND_ONCE)
|
||||||
|
{ return MIG_SERVER_DIED; }
|
||||||
|
else
|
||||||
|
{ return MIG_REPLY_MISMATCH; }
|
||||||
|
}
|
||||||
|
|
||||||
|
#if __MigTypeCheck
|
||||||
|
msgh_size = Out0P->Head.msgh_size;
|
||||||
|
|
||||||
|
if ((Out0P->Head.msgh_bits & MACH_MSGH_BITS_COMPLEX) ||
|
||||||
|
((msgh_size > (mach_msg_size_t)sizeof(__Reply) || msgh_size < (mach_msg_size_t)(sizeof(__Reply) - 896)) &&
|
||||||
|
(msgh_size != (mach_msg_size_t)sizeof(mig_reply_error_t) ||
|
||||||
|
Out0P->RetCode == KERN_SUCCESS)))
|
||||||
|
{ return MIG_TYPE_ERROR ; }
|
||||||
|
#endif /* __MigTypeCheck */
|
||||||
|
|
||||||
|
if (Out0P->RetCode != KERN_SUCCESS) {
|
||||||
|
return ((mig_reply_error_t *)Out0P)->RetCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if __MigTypeCheck
|
||||||
|
if ( Out0P->new_stateCnt > 224 )
|
||||||
|
return MIG_TYPE_ERROR;
|
||||||
|
if (((msgh_size - (mach_msg_size_t)(sizeof(__Reply) - 896)) / 4< Out0P->new_stateCnt) ||
|
||||||
|
(msgh_size != (mach_msg_size_t)(sizeof(__Reply) - 896) + Out0P->new_stateCnt * 4))
|
||||||
|
{ return MIG_TYPE_ERROR ; }
|
||||||
|
#endif /* __MigTypeCheck */
|
||||||
|
|
||||||
|
return MACH_MSG_SUCCESS;
|
||||||
|
}
|
||||||
|
#endif /* !defined(__MIG_check__Reply__mach_exception_raise_state_identity_t__defined) */
|
||||||
|
#endif /* __MIG_check__Reply__mach_exc_subsystem__ */
|
||||||
|
#endif /* ( __MigTypeCheck ) */
|
||||||
|
|
||||||
|
|
||||||
|
/* Routine mach_exception_raise_state_identity */
|
||||||
|
mig_external kern_return_t mach_exception_raise_state_identity
|
||||||
|
(
|
||||||
|
mach_port_t exception_port,
|
||||||
|
mach_port_t thread,
|
||||||
|
mach_port_t task,
|
||||||
|
exception_type_t exception,
|
||||||
|
mach_exception_data_t code,
|
||||||
|
mach_msg_type_number_t codeCnt,
|
||||||
|
int *flavor,
|
||||||
|
thread_state_t old_state,
|
||||||
|
mach_msg_type_number_t old_stateCnt,
|
||||||
|
thread_state_t new_state,
|
||||||
|
mach_msg_type_number_t *new_stateCnt
|
||||||
|
)
|
||||||
|
{
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
/* start of the kernel processed data */
|
||||||
|
mach_msg_body_t msgh_body;
|
||||||
|
mach_msg_port_descriptor_t thread;
|
||||||
|
mach_msg_port_descriptor_t task;
|
||||||
|
/* end of the kernel processed data */
|
||||||
|
NDR_record_t NDR;
|
||||||
|
exception_type_t exception;
|
||||||
|
mach_msg_type_number_t codeCnt;
|
||||||
|
int64_t code[2];
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t old_stateCnt;
|
||||||
|
natural_t old_state[224];
|
||||||
|
} Request;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t new_stateCnt;
|
||||||
|
natural_t new_state[224];
|
||||||
|
mach_msg_trailer_t trailer;
|
||||||
|
} Reply;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack(4)
|
||||||
|
#endif
|
||||||
|
typedef struct {
|
||||||
|
mach_msg_header_t Head;
|
||||||
|
NDR_record_t NDR;
|
||||||
|
kern_return_t RetCode;
|
||||||
|
int flavor;
|
||||||
|
mach_msg_type_number_t new_stateCnt;
|
||||||
|
natural_t new_state[224];
|
||||||
|
} __Reply;
|
||||||
|
#ifdef __MigPackStructs
|
||||||
|
#pragma pack()
|
||||||
|
#endif
|
||||||
|
/*
|
||||||
|
* typedef struct {
|
||||||
|
* mach_msg_header_t Head;
|
||||||
|
* NDR_record_t NDR;
|
||||||
|
* kern_return_t RetCode;
|
||||||
|
* } mig_reply_error_t;
|
||||||
|
*/
|
||||||
|
|
||||||
|
union {
|
||||||
|
Request In;
|
||||||
|
Reply Out;
|
||||||
|
} Mess;
|
||||||
|
|
||||||
|
Request *InP = &Mess.In;
|
||||||
|
Reply *Out0P = &Mess.Out;
|
||||||
|
|
||||||
|
mach_msg_return_t msg_result;
|
||||||
|
unsigned int msgh_size;
|
||||||
|
unsigned int msgh_size_delta;
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef __MIG_check__Reply__mach_exception_raise_state_identity_t__defined
|
||||||
|
kern_return_t check_result;
|
||||||
|
#endif /* __MIG_check__Reply__mach_exception_raise_state_identity_t__defined */
|
||||||
|
|
||||||
|
__DeclareSendRpc(2407, "mach_exception_raise_state_identity")
|
||||||
|
|
||||||
|
#if UseStaticTemplates
|
||||||
|
const static mach_msg_port_descriptor_t threadTemplate = {
|
||||||
|
/* name = */ MACH_PORT_NULL,
|
||||||
|
/* pad1 = */ 0,
|
||||||
|
/* pad2 = */ 0,
|
||||||
|
/* disp = */ 19,
|
||||||
|
/* type = */ MACH_MSG_PORT_DESCRIPTOR,
|
||||||
|
};
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
#if UseStaticTemplates
|
||||||
|
const static mach_msg_port_descriptor_t taskTemplate = {
|
||||||
|
/* name = */ MACH_PORT_NULL,
|
||||||
|
/* pad1 = */ 0,
|
||||||
|
/* pad2 = */ 0,
|
||||||
|
/* disp = */ 19,
|
||||||
|
/* type = */ MACH_MSG_PORT_DESCRIPTOR,
|
||||||
|
};
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
InP->msgh_body.msgh_descriptor_count = 2;
|
||||||
|
#if UseStaticTemplates
|
||||||
|
InP->thread = threadTemplate;
|
||||||
|
InP->thread.name = thread;
|
||||||
|
#else /* UseStaticTemplates */
|
||||||
|
InP->thread.name = thread;
|
||||||
|
InP->thread.disposition = 19;
|
||||||
|
InP->thread.type = MACH_MSG_PORT_DESCRIPTOR;
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
#if UseStaticTemplates
|
||||||
|
InP->task = taskTemplate;
|
||||||
|
InP->task.name = task;
|
||||||
|
#else /* UseStaticTemplates */
|
||||||
|
InP->task.name = task;
|
||||||
|
InP->task.disposition = 19;
|
||||||
|
InP->task.type = MACH_MSG_PORT_DESCRIPTOR;
|
||||||
|
#endif /* UseStaticTemplates */
|
||||||
|
|
||||||
|
InP->NDR = NDR_record;
|
||||||
|
|
||||||
|
InP->exception = exception;
|
||||||
|
|
||||||
|
if (codeCnt > 2) {
|
||||||
|
{ return MIG_ARRAY_TOO_LARGE; }
|
||||||
|
}
|
||||||
|
(void)memcpy((char *) InP->code, (const char *) code, 8 * codeCnt);
|
||||||
|
|
||||||
|
InP->codeCnt = codeCnt;
|
||||||
|
|
||||||
|
msgh_size_delta = (8 * codeCnt);
|
||||||
|
msgh_size = (mach_msg_size_t)(sizeof(Request) - 912) + msgh_size_delta;
|
||||||
|
InP = (Request *) ((pointer_t) InP + msgh_size_delta - 16);
|
||||||
|
|
||||||
|
InP->flavor = *flavor;
|
||||||
|
|
||||||
|
if (old_stateCnt > 224) {
|
||||||
|
{ return MIG_ARRAY_TOO_LARGE; }
|
||||||
|
}
|
||||||
|
(void)memcpy((char *) InP->old_state, (const char *) old_state, 4 * old_stateCnt);
|
||||||
|
|
||||||
|
InP->old_stateCnt = old_stateCnt;
|
||||||
|
|
||||||
|
msgh_size += (4 * old_stateCnt);
|
||||||
|
InP = &Mess.In;
|
||||||
|
InP->Head.msgh_bits = MACH_MSGH_BITS_COMPLEX|
|
||||||
|
MACH_MSGH_BITS(19, MACH_MSG_TYPE_MAKE_SEND_ONCE);
|
||||||
|
/* msgh_size passed as argument */
|
||||||
|
InP->Head.msgh_request_port = exception_port;
|
||||||
|
InP->Head.msgh_reply_port = mig_get_reply_port();
|
||||||
|
InP->Head.msgh_id = 2407;
|
||||||
|
|
||||||
|
/* BEGIN VOUCHER CODE */
|
||||||
|
|
||||||
|
#ifdef USING_VOUCHERS
|
||||||
|
if (voucher_mach_msg_set != NULL) {
|
||||||
|
voucher_mach_msg_set(&InP->Head);
|
||||||
|
}
|
||||||
|
#endif // USING_VOUCHERS
|
||||||
|
|
||||||
|
/* END VOUCHER CODE */
|
||||||
|
|
||||||
|
__BeforeSendRpc(2407, "mach_exception_raise_state_identity")
|
||||||
|
msg_result = mach_msg(&InP->Head, MACH_SEND_MSG|MACH_RCV_MSG|MACH_MSG_OPTION_NONE, msgh_size, (mach_msg_size_t)sizeof(Reply), InP->Head.msgh_reply_port, MACH_MSG_TIMEOUT_NONE, MACH_PORT_NULL);
|
||||||
|
__AfterSendRpc(2407, "mach_exception_raise_state_identity")
|
||||||
|
if (msg_result != MACH_MSG_SUCCESS) {
|
||||||
|
__MachMsgErrorWithoutTimeout(msg_result);
|
||||||
|
{ return msg_result; }
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#if defined(__MIG_check__Reply__mach_exception_raise_state_identity_t__defined)
|
||||||
|
check_result = __MIG_check__Reply__mach_exception_raise_state_identity_t((__Reply__mach_exception_raise_state_identity_t *)Out0P);
|
||||||
|
if (check_result != MACH_MSG_SUCCESS)
|
||||||
|
{ return check_result; }
|
||||||
|
#endif /* defined(__MIG_check__Reply__mach_exception_raise_state_identity_t__defined) */
|
||||||
|
|
||||||
|
*flavor = Out0P->flavor;
|
||||||
|
|
||||||
|
if (Out0P->new_stateCnt > 224) {
|
||||||
|
(void)memcpy((char *) new_state, (const char *) Out0P->new_state, 4 * 224);
|
||||||
|
*new_stateCnt = Out0P->new_stateCnt;
|
||||||
|
{ return MIG_ARRAY_TOO_LARGE; }
|
||||||
|
}
|
||||||
|
(void)memcpy((char *) new_state, (const char *) Out0P->new_state, 4 * Out0P->new_stateCnt);
|
||||||
|
|
||||||
|
*new_stateCnt = Out0P->new_stateCnt;
|
||||||
|
|
||||||
|
return KERN_SUCCESS;
|
||||||
|
}
|
57
vendor/github.com/derekparker/delve/pkg/proc/mem.go
generated
vendored
Normal file
57
vendor/github.com/derekparker/delve/pkg/proc/mem.go
generated
vendored
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
const cacheEnabled = true
|
||||||
|
|
||||||
|
type memoryReadWriter interface {
|
||||||
|
readMemory(addr uintptr, size int) (data []byte, err error)
|
||||||
|
writeMemory(addr uintptr, data []byte) (written int, err error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type memCache struct {
|
||||||
|
cacheAddr uintptr
|
||||||
|
cache []byte
|
||||||
|
mem memoryReadWriter
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *memCache) contains(addr uintptr, size int) bool {
|
||||||
|
return addr >= m.cacheAddr && addr <= (m.cacheAddr+uintptr(len(m.cache)-size))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *memCache) readMemory(addr uintptr, size int) (data []byte, err error) {
|
||||||
|
if m.contains(addr, size) {
|
||||||
|
d := make([]byte, size)
|
||||||
|
copy(d, m.cache[addr-m.cacheAddr:])
|
||||||
|
return d, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return m.mem.readMemory(addr, size)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *memCache) writeMemory(addr uintptr, data []byte) (written int, err error) {
|
||||||
|
return m.mem.writeMemory(addr, data)
|
||||||
|
}
|
||||||
|
|
||||||
|
func cacheMemory(mem memoryReadWriter, addr uintptr, size int) memoryReadWriter {
|
||||||
|
if !cacheEnabled {
|
||||||
|
return mem
|
||||||
|
}
|
||||||
|
if size <= 0 {
|
||||||
|
return mem
|
||||||
|
}
|
||||||
|
if cacheMem, isCache := mem.(*memCache); isCache {
|
||||||
|
if cacheMem.contains(addr, size) {
|
||||||
|
return mem
|
||||||
|
} else {
|
||||||
|
cache, err := cacheMem.mem.readMemory(addr, size)
|
||||||
|
if err != nil {
|
||||||
|
return mem
|
||||||
|
}
|
||||||
|
return &memCache{addr, cache, mem}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cache, err := mem.readMemory(addr, size)
|
||||||
|
if err != nil {
|
||||||
|
return mem
|
||||||
|
}
|
||||||
|
return &memCache{addr, cache, mem}
|
||||||
|
}
|
189
vendor/github.com/derekparker/delve/pkg/proc/moduledata.go
generated
vendored
Normal file
189
vendor/github.com/derekparker/delve/pkg/proc/moduledata.go
generated
vendored
Normal file
@ -0,0 +1,189 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"go/constant"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
// delve counterpart to runtime.moduledata
|
||||||
|
type moduleData struct {
|
||||||
|
types, etypes uintptr
|
||||||
|
typemapVar *Variable
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) loadModuleData() (err error) {
|
||||||
|
dbp.loadModuleDataOnce.Do(func() {
|
||||||
|
scope := &EvalScope{Thread: dbp.currentThread, PC: 0, CFA: 0}
|
||||||
|
var md *Variable
|
||||||
|
md, err = scope.packageVarAddr("runtime.firstmoduledata")
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for md.Addr != 0 {
|
||||||
|
var typesVar, etypesVar, nextVar, typemapVar *Variable
|
||||||
|
var types, etypes uint64
|
||||||
|
|
||||||
|
if typesVar, err = md.structMember("types"); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if etypesVar, err = md.structMember("etypes"); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if nextVar, err = md.structMember("next"); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if typemapVar, err = md.structMember("typemap"); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if types, err = typesVar.asUint(); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if etypes, err = etypesVar.asUint(); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
dbp.moduleData = append(dbp.moduleData, moduleData{uintptr(types), uintptr(etypes), typemapVar})
|
||||||
|
|
||||||
|
md = nextVar.maybeDereference()
|
||||||
|
if md.Unreadable != nil {
|
||||||
|
err = md.Unreadable
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) resolveTypeOff(typeAddr uintptr, off uintptr) (*Variable, error) {
|
||||||
|
// See runtime.(*_type).typeOff in $GOROOT/src/runtime/type.go
|
||||||
|
if err := dbp.loadModuleData(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var md *moduleData
|
||||||
|
for i := range dbp.moduleData {
|
||||||
|
if typeAddr >= dbp.moduleData[i].types && typeAddr < dbp.moduleData[i].etypes {
|
||||||
|
md = &dbp.moduleData[i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
rtyp, err := dbp.findType("runtime._type")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if md == nil {
|
||||||
|
v, err := dbp.reflectOffsMapAccess(off)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
v.loadValue(LoadConfig{false, 1, 0, 0, -1})
|
||||||
|
addr, _ := constant.Int64Val(v.Value)
|
||||||
|
return v.newVariable(v.Name, uintptr(addr), rtyp), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if t, _ := md.typemapVar.mapAccess(newConstant(constant.MakeUint64(uint64(off)), dbp.currentThread)); t != nil {
|
||||||
|
return t, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
res := md.types + uintptr(off)
|
||||||
|
|
||||||
|
return dbp.currentThread.newVariable("", res, rtyp), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) resolveNameOff(typeAddr uintptr, off uintptr) (name, tag string, pkgpathoff int32, err error) {
|
||||||
|
// See runtime.resolveNameOff in $GOROOT/src/runtime/type.go
|
||||||
|
if err = dbp.loadModuleData(); err != nil {
|
||||||
|
return "", "", 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, md := range dbp.moduleData {
|
||||||
|
if typeAddr >= md.types && typeAddr < md.etypes {
|
||||||
|
return dbp.loadName(md.types + off)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
v, err := dbp.reflectOffsMapAccess(off)
|
||||||
|
if err != nil {
|
||||||
|
return "", "", 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
resv := v.maybeDereference()
|
||||||
|
if resv.Unreadable != nil {
|
||||||
|
return "", "", 0, resv.Unreadable
|
||||||
|
}
|
||||||
|
|
||||||
|
return dbp.loadName(resv.Addr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) reflectOffsMapAccess(off uintptr) (*Variable, error) {
|
||||||
|
scope := &EvalScope{Thread: dbp.currentThread, PC: 0, CFA: 0}
|
||||||
|
reflectOffs, err := scope.packageVarAddr("runtime.reflectOffs")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
reflectOffsm, err := reflectOffs.structMember("m")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return reflectOffsm.mapAccess(newConstant(constant.MakeUint64(uint64(off)), dbp.currentThread))
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
// flags for the name struct (see 'type name struct' in $GOROOT/src/reflect/type.go)
|
||||||
|
nameflagExported = 1 << 0
|
||||||
|
nameflagHasTag = 1 << 1
|
||||||
|
nameflagHasPkg = 1 << 2
|
||||||
|
)
|
||||||
|
|
||||||
|
func (dbp *Process) loadName(addr uintptr) (name, tag string, pkgpathoff int32, err error) {
|
||||||
|
off := addr
|
||||||
|
namedata, err := dbp.currentThread.readMemory(off, 3)
|
||||||
|
off += 3
|
||||||
|
if err != nil {
|
||||||
|
return "", "", 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
namelen := uint16(namedata[1]<<8) | uint16(namedata[2])
|
||||||
|
|
||||||
|
rawstr, err := dbp.currentThread.readMemory(off, int(namelen))
|
||||||
|
off += uintptr(namelen)
|
||||||
|
if err != nil {
|
||||||
|
return "", "", 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
name = string(rawstr)
|
||||||
|
|
||||||
|
if namedata[0]&nameflagHasTag != 0 {
|
||||||
|
taglendata, err := dbp.currentThread.readMemory(off, 2)
|
||||||
|
off += 2
|
||||||
|
if err != nil {
|
||||||
|
return "", "", 0, err
|
||||||
|
}
|
||||||
|
taglen := uint16(taglendata[0]<<8) | uint16(taglendata[1])
|
||||||
|
|
||||||
|
rawstr, err := dbp.currentThread.readMemory(off, int(taglen))
|
||||||
|
off += uintptr(taglen)
|
||||||
|
if err != nil {
|
||||||
|
return "", "", 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
tag = string(rawstr)
|
||||||
|
}
|
||||||
|
|
||||||
|
if namedata[0]&nameflagHasPkg != 0 {
|
||||||
|
pkgdata, err := dbp.currentThread.readMemory(off, 4)
|
||||||
|
if err != nil {
|
||||||
|
return "", "", 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// see func pkgPath in $GOROOT/src/reflect/type.go
|
||||||
|
copy((*[4]byte)(unsafe.Pointer(&pkgpathoff))[:], pkgdata)
|
||||||
|
}
|
||||||
|
|
||||||
|
return name, tag, pkgpathoff, nil
|
||||||
|
}
|
986
vendor/github.com/derekparker/delve/pkg/proc/proc.go
generated
vendored
Normal file
986
vendor/github.com/derekparker/delve/pkg/proc/proc.go
generated
vendored
Normal file
@ -0,0 +1,986 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"debug/gosym"
|
||||||
|
"encoding/binary"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"go/ast"
|
||||||
|
"go/constant"
|
||||||
|
"go/token"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"runtime"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/frame"
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/line"
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/reader"
|
||||||
|
"golang.org/x/debug/dwarf"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Process represents all of the information the debugger
|
||||||
|
// is holding onto regarding the process we are debugging.
|
||||||
|
type Process struct {
|
||||||
|
pid int // Process Pid
|
||||||
|
Process *os.Process // Pointer to process struct for the actual process we are debugging
|
||||||
|
lastModified time.Time // Time the executable of this process was last modified
|
||||||
|
|
||||||
|
// Breakpoint table, holds information on breakpoints.
|
||||||
|
// Maps instruction address to Breakpoint struct.
|
||||||
|
breakpoints map[uint64]*Breakpoint
|
||||||
|
|
||||||
|
// List of threads mapped as such: pid -> *Thread
|
||||||
|
threads map[int]*Thread
|
||||||
|
|
||||||
|
// Active thread
|
||||||
|
currentThread *Thread
|
||||||
|
|
||||||
|
// Goroutine that will be used by default to set breakpoint, eval variables, etc...
|
||||||
|
// Normally selectedGoroutine is currentThread.GetG, it will not be only if SwitchGoroutine is called with a goroutine that isn't attached to a thread
|
||||||
|
selectedGoroutine *G
|
||||||
|
|
||||||
|
// Maps package names to package paths, needed to lookup types inside DWARF info
|
||||||
|
packageMap map[string]string
|
||||||
|
|
||||||
|
allGCache []*G
|
||||||
|
dwarf *dwarf.Data
|
||||||
|
goSymTable *gosym.Table
|
||||||
|
frameEntries frame.FrameDescriptionEntries
|
||||||
|
lineInfo line.DebugLines
|
||||||
|
os *OSProcessDetails
|
||||||
|
arch Arch
|
||||||
|
breakpointIDCounter int
|
||||||
|
internalBreakpointIDCounter int
|
||||||
|
firstStart bool
|
||||||
|
halt bool
|
||||||
|
exited bool
|
||||||
|
ptraceChan chan func()
|
||||||
|
ptraceDoneChan chan interface{}
|
||||||
|
types map[string]dwarf.Offset
|
||||||
|
functions []functionDebugInfo
|
||||||
|
|
||||||
|
loadModuleDataOnce sync.Once
|
||||||
|
moduleData []moduleData
|
||||||
|
nameOfRuntimeType map[uintptr]nameOfRuntimeTypeEntry
|
||||||
|
}
|
||||||
|
|
||||||
|
type functionDebugInfo struct {
|
||||||
|
lowpc, highpc uint64
|
||||||
|
offset dwarf.Offset
|
||||||
|
}
|
||||||
|
|
||||||
|
var NotExecutableErr = errors.New("not an executable file")
|
||||||
|
|
||||||
|
// New returns an initialized Process struct. Before returning,
|
||||||
|
// it will also launch a goroutine in order to handle ptrace(2)
|
||||||
|
// functions. For more information, see the documentation on
|
||||||
|
// `handlePtraceFuncs`.
|
||||||
|
func New(pid int) *Process {
|
||||||
|
dbp := &Process{
|
||||||
|
pid: pid,
|
||||||
|
threads: make(map[int]*Thread),
|
||||||
|
breakpoints: make(map[uint64]*Breakpoint),
|
||||||
|
firstStart: true,
|
||||||
|
os: new(OSProcessDetails),
|
||||||
|
ptraceChan: make(chan func()),
|
||||||
|
ptraceDoneChan: make(chan interface{}),
|
||||||
|
nameOfRuntimeType: make(map[uintptr]nameOfRuntimeTypeEntry),
|
||||||
|
}
|
||||||
|
// TODO: find better way to determine proc arch (perhaps use executable file info)
|
||||||
|
switch runtime.GOARCH {
|
||||||
|
case "amd64":
|
||||||
|
dbp.arch = AMD64Arch()
|
||||||
|
}
|
||||||
|
go dbp.handlePtraceFuncs()
|
||||||
|
return dbp
|
||||||
|
}
|
||||||
|
|
||||||
|
// ProcessExitedError indicates that the process has exited and contains both
|
||||||
|
// process id and exit status.
|
||||||
|
type ProcessExitedError struct {
|
||||||
|
Pid int
|
||||||
|
Status int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pe ProcessExitedError) Error() string {
|
||||||
|
return fmt.Sprintf("Process %d has exited with status %d", pe.Pid, pe.Status)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Detach from the process being debugged, optionally killing it.
|
||||||
|
func (dbp *Process) Detach(kill bool) (err error) {
|
||||||
|
if dbp.exited {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if dbp.Running() {
|
||||||
|
if err = dbp.Halt(); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !kill {
|
||||||
|
// Clean up any breakpoints we've set.
|
||||||
|
for _, bp := range dbp.breakpoints {
|
||||||
|
if bp != nil {
|
||||||
|
_, err := dbp.ClearBreakpoint(bp.Addr)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
dbp.execPtraceFunc(func() {
|
||||||
|
err = PtraceDetach(dbp.pid, 0)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if kill {
|
||||||
|
err = killProcess(dbp.pid)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Exited returns whether the debugged
|
||||||
|
// process has exited.
|
||||||
|
func (dbp *Process) Exited() bool {
|
||||||
|
return dbp.exited
|
||||||
|
}
|
||||||
|
|
||||||
|
// Running returns whether the debugged
|
||||||
|
// process is currently executing.
|
||||||
|
func (dbp *Process) Running() bool {
|
||||||
|
for _, th := range dbp.threads {
|
||||||
|
if th.running {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) LastModified() time.Time {
|
||||||
|
return dbp.lastModified
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) Pid() int {
|
||||||
|
return dbp.pid
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) SelectedGoroutine() *G {
|
||||||
|
return dbp.selectedGoroutine
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) Threads() map[int]*Thread {
|
||||||
|
return dbp.threads
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) CurrentThread() *Thread {
|
||||||
|
return dbp.currentThread
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) Breakpoints() map[uint64]*Breakpoint {
|
||||||
|
return dbp.breakpoints
|
||||||
|
}
|
||||||
|
|
||||||
|
// LoadInformation finds the executable and then uses it
|
||||||
|
// to parse the following information:
|
||||||
|
// * Dwarf .debug_frame section
|
||||||
|
// * Dwarf .debug_line section
|
||||||
|
// * Go symbol table.
|
||||||
|
func (dbp *Process) LoadInformation(path string) error {
|
||||||
|
var wg sync.WaitGroup
|
||||||
|
|
||||||
|
exe, path, err := dbp.findExecutable(path)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
fi, err := os.Stat(path)
|
||||||
|
if err == nil {
|
||||||
|
dbp.lastModified = fi.ModTime()
|
||||||
|
}
|
||||||
|
|
||||||
|
wg.Add(5)
|
||||||
|
go dbp.loadProcessInformation(&wg)
|
||||||
|
go dbp.parseDebugFrame(exe, &wg)
|
||||||
|
go dbp.obtainGoSymbols(exe, &wg)
|
||||||
|
go dbp.parseDebugLineInfo(exe, &wg)
|
||||||
|
go dbp.loadDebugInfoMaps(&wg)
|
||||||
|
wg.Wait()
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FindFileLocation returns the PC for a given file:line.
|
||||||
|
// Assumes that `file` is normailzed to lower case and '/' on Windows.
|
||||||
|
func (dbp *Process) FindFileLocation(fileName string, lineno int) (uint64, error) {
|
||||||
|
pc, fn, err := dbp.goSymTable.LineToPC(fileName, lineno)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
if fn.Entry == pc {
|
||||||
|
pc, _ = dbp.FirstPCAfterPrologue(fn, true)
|
||||||
|
}
|
||||||
|
return pc, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FindFunctionLocation finds address of a function's line
|
||||||
|
// If firstLine == true is passed FindFunctionLocation will attempt to find the first line of the function
|
||||||
|
// If lineOffset is passed FindFunctionLocation will return the address of that line
|
||||||
|
// Pass lineOffset == 0 and firstLine == false if you want the address for the function's entry point
|
||||||
|
// Note that setting breakpoints at that address will cause surprising behavior:
|
||||||
|
// https://github.com/derekparker/delve/issues/170
|
||||||
|
func (dbp *Process) FindFunctionLocation(funcName string, firstLine bool, lineOffset int) (uint64, error) {
|
||||||
|
origfn := dbp.goSymTable.LookupFunc(funcName)
|
||||||
|
if origfn == nil {
|
||||||
|
return 0, fmt.Errorf("Could not find function %s\n", funcName)
|
||||||
|
}
|
||||||
|
|
||||||
|
if firstLine {
|
||||||
|
return dbp.FirstPCAfterPrologue(origfn, false)
|
||||||
|
} else if lineOffset > 0 {
|
||||||
|
filename, lineno, _ := dbp.goSymTable.PCToLine(origfn.Entry)
|
||||||
|
breakAddr, _, err := dbp.goSymTable.LineToPC(filename, lineno+lineOffset)
|
||||||
|
return breakAddr, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return origfn.Entry, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// CurrentLocation returns the location of the current thread.
|
||||||
|
func (dbp *Process) CurrentLocation() (*Location, error) {
|
||||||
|
return dbp.currentThread.Location()
|
||||||
|
}
|
||||||
|
|
||||||
|
// RequestManualStop sets the `halt` flag and
|
||||||
|
// sends SIGSTOP to all threads.
|
||||||
|
func (dbp *Process) RequestManualStop() error {
|
||||||
|
if dbp.exited {
|
||||||
|
return &ProcessExitedError{}
|
||||||
|
}
|
||||||
|
dbp.halt = true
|
||||||
|
return dbp.requestManualStop()
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetBreakpoint sets a breakpoint at addr, and stores it in the process wide
|
||||||
|
// break point table. Setting a break point must be thread specific due to
|
||||||
|
// ptrace actions needing the thread to be in a signal-delivery-stop.
|
||||||
|
func (dbp *Process) SetBreakpoint(addr uint64, kind BreakpointKind, cond ast.Expr) (*Breakpoint, error) {
|
||||||
|
tid := dbp.currentThread.ID
|
||||||
|
|
||||||
|
if bp, ok := dbp.FindBreakpoint(addr); ok {
|
||||||
|
return nil, BreakpointExistsError{bp.File, bp.Line, bp.Addr}
|
||||||
|
}
|
||||||
|
|
||||||
|
f, l, fn := dbp.goSymTable.PCToLine(uint64(addr))
|
||||||
|
if fn == nil {
|
||||||
|
return nil, InvalidAddressError{address: addr}
|
||||||
|
}
|
||||||
|
|
||||||
|
newBreakpoint := &Breakpoint{
|
||||||
|
FunctionName: fn.Name,
|
||||||
|
File: f,
|
||||||
|
Line: l,
|
||||||
|
Addr: addr,
|
||||||
|
Kind: kind,
|
||||||
|
Cond: cond,
|
||||||
|
HitCount: map[int]uint64{},
|
||||||
|
}
|
||||||
|
|
||||||
|
if kind != UserBreakpoint {
|
||||||
|
dbp.internalBreakpointIDCounter++
|
||||||
|
newBreakpoint.ID = dbp.internalBreakpointIDCounter
|
||||||
|
} else {
|
||||||
|
dbp.breakpointIDCounter++
|
||||||
|
newBreakpoint.ID = dbp.breakpointIDCounter
|
||||||
|
}
|
||||||
|
|
||||||
|
thread := dbp.threads[tid]
|
||||||
|
originalData, err := thread.readMemory(uintptr(addr), dbp.arch.BreakpointSize())
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if err := dbp.writeSoftwareBreakpoint(thread, addr); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
newBreakpoint.OriginalData = originalData
|
||||||
|
dbp.breakpoints[addr] = newBreakpoint
|
||||||
|
|
||||||
|
return newBreakpoint, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ClearBreakpoint clears the breakpoint at addr.
|
||||||
|
func (dbp *Process) ClearBreakpoint(addr uint64) (*Breakpoint, error) {
|
||||||
|
if dbp.exited {
|
||||||
|
return nil, &ProcessExitedError{}
|
||||||
|
}
|
||||||
|
bp, ok := dbp.FindBreakpoint(addr)
|
||||||
|
if !ok {
|
||||||
|
return nil, NoBreakpointError{addr: addr}
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := bp.Clear(dbp.currentThread); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
delete(dbp.breakpoints, addr)
|
||||||
|
|
||||||
|
return bp, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Status returns the status of the current main thread context.
|
||||||
|
func (dbp *Process) Status() *WaitStatus {
|
||||||
|
return dbp.currentThread.Status
|
||||||
|
}
|
||||||
|
|
||||||
|
// Next continues execution until the next source line.
|
||||||
|
func (dbp *Process) Next() (err error) {
|
||||||
|
if dbp.exited {
|
||||||
|
return &ProcessExitedError{}
|
||||||
|
}
|
||||||
|
for i := range dbp.breakpoints {
|
||||||
|
if dbp.breakpoints[i].Internal() {
|
||||||
|
return fmt.Errorf("next while nexting")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = dbp.next(false); err != nil {
|
||||||
|
switch err.(type) {
|
||||||
|
case ThreadBlockedError, NoReturnAddr: // Noop
|
||||||
|
default:
|
||||||
|
dbp.ClearInternalBreakpoints()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return dbp.Continue()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Continue continues execution of the debugged
|
||||||
|
// process. It will continue until it hits a breakpoint
|
||||||
|
// or is otherwise stopped.
|
||||||
|
func (dbp *Process) Continue() error {
|
||||||
|
if dbp.exited {
|
||||||
|
return &ProcessExitedError{}
|
||||||
|
}
|
||||||
|
for {
|
||||||
|
if err := dbp.resume(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
dbp.allGCache = nil
|
||||||
|
for _, th := range dbp.threads {
|
||||||
|
th.clearBreakpointState()
|
||||||
|
}
|
||||||
|
|
||||||
|
trapthread, err := dbp.trapWait(-1)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := dbp.Halt(); err != nil {
|
||||||
|
return dbp.exitGuard(err)
|
||||||
|
}
|
||||||
|
if err := dbp.setCurrentBreakpoints(trapthread); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := dbp.pickCurrentThread(trapthread); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
switch {
|
||||||
|
case dbp.currentThread.CurrentBreakpoint == nil:
|
||||||
|
// runtime.Breakpoint or manual stop
|
||||||
|
if dbp.currentThread.onRuntimeBreakpoint() {
|
||||||
|
// Single-step current thread until we exit runtime.breakpoint and
|
||||||
|
// runtime.Breakpoint.
|
||||||
|
// On go < 1.8 it was sufficient to single-step twice on go1.8 a change
|
||||||
|
// to the compiler requires 4 steps.
|
||||||
|
for {
|
||||||
|
if err = dbp.currentThread.StepInstruction(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
loc, err := dbp.currentThread.Location()
|
||||||
|
if err != nil || loc.Fn == nil || (loc.Fn.Name != "runtime.breakpoint" && loc.Fn.Name != "runtime.Breakpoint") {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return dbp.conditionErrors()
|
||||||
|
case dbp.currentThread.onTriggeredInternalBreakpoint():
|
||||||
|
if dbp.currentThread.CurrentBreakpoint.Kind == StepBreakpoint {
|
||||||
|
// See description of proc.(*Process).next for the meaning of StepBreakpoints
|
||||||
|
if err := dbp.conditionErrors(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
pc, err := dbp.currentThread.PC()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
text, err := dbp.currentThread.Disassemble(pc, pc+maxInstructionLength, true)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
// here we either set a breakpoint into the destination of the CALL
|
||||||
|
// instruction or we determined that the called function is hidden,
|
||||||
|
// either way we need to resume execution
|
||||||
|
if err = dbp.setStepIntoBreakpoint(text, sameGoroutineCondition(dbp.selectedGoroutine)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if err := dbp.ClearInternalBreakpoints(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return dbp.conditionErrors()
|
||||||
|
}
|
||||||
|
case dbp.currentThread.onTriggeredBreakpoint():
|
||||||
|
onNextGoroutine, err := dbp.currentThread.onNextGoroutine()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if onNextGoroutine {
|
||||||
|
err := dbp.ClearInternalBreakpoints()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return dbp.conditionErrors()
|
||||||
|
default:
|
||||||
|
// not a manual stop, not on runtime.Breakpoint, not on a breakpoint, just repeat
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) conditionErrors() error {
|
||||||
|
var condErr error
|
||||||
|
for _, th := range dbp.threads {
|
||||||
|
if th.CurrentBreakpoint != nil && th.BreakpointConditionError != nil {
|
||||||
|
if condErr == nil {
|
||||||
|
condErr = th.BreakpointConditionError
|
||||||
|
} else {
|
||||||
|
return fmt.Errorf("multiple errors evaluating conditions")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return condErr
|
||||||
|
}
|
||||||
|
|
||||||
|
// pick a new dbp.currentThread, with the following priority:
|
||||||
|
// - a thread with onTriggeredInternalBreakpoint() == true
|
||||||
|
// - a thread with onTriggeredBreakpoint() == true (prioritizing trapthread)
|
||||||
|
// - trapthread
|
||||||
|
func (dbp *Process) pickCurrentThread(trapthread *Thread) error {
|
||||||
|
for _, th := range dbp.threads {
|
||||||
|
if th.onTriggeredInternalBreakpoint() {
|
||||||
|
return dbp.SwitchThread(th.ID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if trapthread.onTriggeredBreakpoint() {
|
||||||
|
return dbp.SwitchThread(trapthread.ID)
|
||||||
|
}
|
||||||
|
for _, th := range dbp.threads {
|
||||||
|
if th.onTriggeredBreakpoint() {
|
||||||
|
return dbp.SwitchThread(th.ID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return dbp.SwitchThread(trapthread.ID)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Step will continue until another source line is reached.
|
||||||
|
// Will step into functions.
|
||||||
|
func (dbp *Process) Step() (err error) {
|
||||||
|
if dbp.exited {
|
||||||
|
return &ProcessExitedError{}
|
||||||
|
}
|
||||||
|
for i := range dbp.breakpoints {
|
||||||
|
if dbp.breakpoints[i].Internal() {
|
||||||
|
return fmt.Errorf("next while nexting")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = dbp.next(true); err != nil {
|
||||||
|
switch err.(type) {
|
||||||
|
case ThreadBlockedError, NoReturnAddr: // Noop
|
||||||
|
default:
|
||||||
|
dbp.ClearInternalBreakpoints()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return dbp.Continue()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns an expression that evaluates to true when the current goroutine is g
|
||||||
|
func sameGoroutineCondition(g *G) ast.Expr {
|
||||||
|
if g == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return &ast.BinaryExpr{
|
||||||
|
Op: token.EQL,
|
||||||
|
X: &ast.SelectorExpr{
|
||||||
|
X: &ast.SelectorExpr{
|
||||||
|
X: &ast.Ident{Name: "runtime"},
|
||||||
|
Sel: &ast.Ident{Name: "curg"},
|
||||||
|
},
|
||||||
|
Sel: &ast.Ident{Name: "goid"},
|
||||||
|
},
|
||||||
|
Y: &ast.BasicLit{Kind: token.INT, Value: strconv.Itoa(g.ID)},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// StepInstruction will continue the current thread for exactly
|
||||||
|
// one instruction. This method affects only the thread
|
||||||
|
// asssociated with the selected goroutine. All other
|
||||||
|
// threads will remain stopped.
|
||||||
|
func (dbp *Process) StepInstruction() (err error) {
|
||||||
|
if dbp.selectedGoroutine == nil {
|
||||||
|
return errors.New("cannot single step: no selected goroutine")
|
||||||
|
}
|
||||||
|
if dbp.selectedGoroutine.thread == nil {
|
||||||
|
// Step called on parked goroutine
|
||||||
|
if _, err := dbp.SetBreakpoint(dbp.selectedGoroutine.PC, NextBreakpoint, sameGoroutineCondition(dbp.selectedGoroutine)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return dbp.Continue()
|
||||||
|
}
|
||||||
|
dbp.allGCache = nil
|
||||||
|
if dbp.exited {
|
||||||
|
return &ProcessExitedError{}
|
||||||
|
}
|
||||||
|
dbp.selectedGoroutine.thread.clearBreakpointState()
|
||||||
|
err = dbp.selectedGoroutine.thread.StepInstruction()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return dbp.selectedGoroutine.thread.SetCurrentBreakpoint()
|
||||||
|
}
|
||||||
|
|
||||||
|
// StepOut will continue until the current goroutine exits the
|
||||||
|
// function currently being executed or a deferred function is executed
|
||||||
|
func (dbp *Process) StepOut() error {
|
||||||
|
cond := sameGoroutineCondition(dbp.selectedGoroutine)
|
||||||
|
|
||||||
|
topframe, err := topframe(dbp.selectedGoroutine, dbp.currentThread)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
pcs := []uint64{}
|
||||||
|
|
||||||
|
var deferpc uint64 = 0
|
||||||
|
if filepath.Ext(topframe.Current.File) == ".go" {
|
||||||
|
if dbp.selectedGoroutine != nil {
|
||||||
|
deferPCEntry := dbp.selectedGoroutine.DeferPC()
|
||||||
|
if deferPCEntry != 0 {
|
||||||
|
_, _, deferfn := dbp.goSymTable.PCToLine(deferPCEntry)
|
||||||
|
deferpc, err = dbp.FirstPCAfterPrologue(deferfn, false)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
pcs = append(pcs, deferpc)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if topframe.Ret == 0 && deferpc == 0 {
|
||||||
|
return errors.New("nothing to stepout to")
|
||||||
|
}
|
||||||
|
|
||||||
|
if deferpc != 0 && deferpc != topframe.Current.PC {
|
||||||
|
bp, err := dbp.SetBreakpoint(deferpc, NextDeferBreakpoint, cond)
|
||||||
|
if err != nil {
|
||||||
|
if _, ok := err.(BreakpointExistsError); !ok {
|
||||||
|
dbp.ClearInternalBreakpoints()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if bp != nil {
|
||||||
|
// For StepOut we do not want to step into the deferred function
|
||||||
|
// when it's called by runtime.deferreturn so we do not populate
|
||||||
|
// DeferReturns.
|
||||||
|
bp.DeferReturns = []uint64{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if topframe.Ret != 0 {
|
||||||
|
if err := dbp.setInternalBreakpoints(topframe.Current.PC, []uint64{topframe.Ret}, NextBreakpoint, cond); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return dbp.Continue()
|
||||||
|
}
|
||||||
|
|
||||||
|
// SwitchThread changes from current thread to the thread specified by `tid`.
|
||||||
|
func (dbp *Process) SwitchThread(tid int) error {
|
||||||
|
if dbp.exited {
|
||||||
|
return &ProcessExitedError{}
|
||||||
|
}
|
||||||
|
if th, ok := dbp.threads[tid]; ok {
|
||||||
|
dbp.currentThread = th
|
||||||
|
dbp.selectedGoroutine, _ = dbp.currentThread.GetG()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return fmt.Errorf("thread %d does not exist", tid)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SwitchGoroutine changes from current thread to the thread
|
||||||
|
// running the specified goroutine.
|
||||||
|
func (dbp *Process) SwitchGoroutine(gid int) error {
|
||||||
|
if dbp.exited {
|
||||||
|
return &ProcessExitedError{}
|
||||||
|
}
|
||||||
|
g, err := dbp.FindGoroutine(gid)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if g == nil {
|
||||||
|
// user specified -1 and selectedGoroutine is nil
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if g.thread != nil {
|
||||||
|
return dbp.SwitchThread(g.thread.ID)
|
||||||
|
}
|
||||||
|
dbp.selectedGoroutine = g
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GoroutinesInfo returns an array of G structures representing the information
|
||||||
|
// Delve cares about from the internal runtime G structure.
|
||||||
|
func (dbp *Process) GoroutinesInfo() ([]*G, error) {
|
||||||
|
if dbp.exited {
|
||||||
|
return nil, &ProcessExitedError{}
|
||||||
|
}
|
||||||
|
if dbp.allGCache != nil {
|
||||||
|
return dbp.allGCache, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
threadg = map[int]*Thread{}
|
||||||
|
allg []*G
|
||||||
|
rdr = dbp.DwarfReader()
|
||||||
|
)
|
||||||
|
|
||||||
|
for i := range dbp.threads {
|
||||||
|
if dbp.threads[i].blocked() {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
g, _ := dbp.threads[i].GetG()
|
||||||
|
if g != nil {
|
||||||
|
threadg[g.ID] = dbp.threads[i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
addr, err := rdr.AddrFor("runtime.allglen")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
allglenBytes, err := dbp.currentThread.readMemory(uintptr(addr), 8)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
allglen := binary.LittleEndian.Uint64(allglenBytes)
|
||||||
|
|
||||||
|
rdr.Seek(0)
|
||||||
|
allgentryaddr, err := rdr.AddrFor("runtime.allgs")
|
||||||
|
if err != nil {
|
||||||
|
// try old name (pre Go 1.6)
|
||||||
|
allgentryaddr, err = rdr.AddrFor("runtime.allg")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
faddr, err := dbp.currentThread.readMemory(uintptr(allgentryaddr), dbp.arch.PtrSize())
|
||||||
|
allgptr := binary.LittleEndian.Uint64(faddr)
|
||||||
|
|
||||||
|
for i := uint64(0); i < allglen; i++ {
|
||||||
|
gvar, err := dbp.currentThread.newGVariable(uintptr(allgptr+(i*uint64(dbp.arch.PtrSize()))), true)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
g, err := gvar.parseG()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if thread, allocated := threadg[g.ID]; allocated {
|
||||||
|
loc, err := thread.Location()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
g.thread = thread
|
||||||
|
// Prefer actual thread location information.
|
||||||
|
g.CurrentLoc = *loc
|
||||||
|
}
|
||||||
|
if g.Status != Gdead {
|
||||||
|
allg = append(allg, g)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
dbp.allGCache = allg
|
||||||
|
return allg, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *G) Thread() *Thread {
|
||||||
|
return g.thread
|
||||||
|
}
|
||||||
|
|
||||||
|
// Halt stops all threads.
|
||||||
|
func (dbp *Process) Halt() (err error) {
|
||||||
|
if dbp.exited {
|
||||||
|
return &ProcessExitedError{}
|
||||||
|
}
|
||||||
|
for _, th := range dbp.threads {
|
||||||
|
if err := th.Halt(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Registers obtains register values from the
|
||||||
|
// "current" thread of the traced process.
|
||||||
|
func (dbp *Process) Registers() (Registers, error) {
|
||||||
|
return dbp.currentThread.Registers(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PC returns the PC of the current thread.
|
||||||
|
func (dbp *Process) PC() (uint64, error) {
|
||||||
|
return dbp.currentThread.PC()
|
||||||
|
}
|
||||||
|
|
||||||
|
// CurrentBreakpoint returns the breakpoint the current thread
|
||||||
|
// is stopped at.
|
||||||
|
func (dbp *Process) CurrentBreakpoint() *Breakpoint {
|
||||||
|
return dbp.currentThread.CurrentBreakpoint
|
||||||
|
}
|
||||||
|
|
||||||
|
// DwarfReader returns a reader for the dwarf data
|
||||||
|
func (dbp *Process) DwarfReader() *reader.Reader {
|
||||||
|
return reader.New(dbp.dwarf)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sources returns list of source files that comprise the debugged binary.
|
||||||
|
func (dbp *Process) Sources() map[string]*gosym.Obj {
|
||||||
|
return dbp.goSymTable.Files
|
||||||
|
}
|
||||||
|
|
||||||
|
// Funcs returns list of functions present in the debugged program.
|
||||||
|
func (dbp *Process) Funcs() []gosym.Func {
|
||||||
|
return dbp.goSymTable.Funcs
|
||||||
|
}
|
||||||
|
|
||||||
|
// Types returns list of types present in the debugged program.
|
||||||
|
func (dbp *Process) Types() ([]string, error) {
|
||||||
|
types := make([]string, 0, len(dbp.types))
|
||||||
|
for k := range dbp.types {
|
||||||
|
types = append(types, k)
|
||||||
|
}
|
||||||
|
return types, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// PCToLine converts an instruction address to a file/line/function.
|
||||||
|
func (dbp *Process) PCToLine(pc uint64) (string, int, *gosym.Func) {
|
||||||
|
return dbp.goSymTable.PCToLine(pc)
|
||||||
|
}
|
||||||
|
|
||||||
|
// FindBreakpointByID finds the breakpoint for the given ID.
|
||||||
|
func (dbp *Process) FindBreakpointByID(id int) (*Breakpoint, bool) {
|
||||||
|
for _, bp := range dbp.breakpoints {
|
||||||
|
if bp.ID == id {
|
||||||
|
return bp, true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
|
||||||
|
// FindBreakpoint finds the breakpoint for the given pc.
|
||||||
|
func (dbp *Process) FindBreakpoint(pc uint64) (*Breakpoint, bool) {
|
||||||
|
// Check to see if address is past the breakpoint, (i.e. breakpoint was hit).
|
||||||
|
if bp, ok := dbp.breakpoints[pc-uint64(dbp.arch.BreakpointSize())]; ok {
|
||||||
|
return bp, true
|
||||||
|
}
|
||||||
|
// Directly use addr to lookup breakpoint.
|
||||||
|
if bp, ok := dbp.breakpoints[pc]; ok {
|
||||||
|
return bp, true
|
||||||
|
}
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns a new Process struct.
|
||||||
|
func initializeDebugProcess(dbp *Process, path string, attach bool) (*Process, error) {
|
||||||
|
if attach {
|
||||||
|
var err error
|
||||||
|
dbp.execPtraceFunc(func() { err = PtraceAttach(dbp.pid) })
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
_, _, err = dbp.wait(dbp.pid, 0)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
proc, err := os.FindProcess(dbp.pid)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
dbp.Process = proc
|
||||||
|
err = dbp.LoadInformation(path)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := dbp.updateThreadList(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
ver, isextld, err := dbp.getGoInformation()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
dbp.arch.SetGStructOffset(ver, isextld)
|
||||||
|
// selectedGoroutine can not be set correctly by the call to updateThreadList
|
||||||
|
// because without calling SetGStructOffset we can not read the G struct of currentThread
|
||||||
|
// but without calling updateThreadList we can not examine memory to determine
|
||||||
|
// the offset of g struct inside TLS
|
||||||
|
dbp.selectedGoroutine, _ = dbp.currentThread.GetG()
|
||||||
|
|
||||||
|
panicpc, err := dbp.FindFunctionLocation("runtime.startpanic", true, 0)
|
||||||
|
if err == nil {
|
||||||
|
bp, err := dbp.SetBreakpoint(panicpc, UserBreakpoint, nil)
|
||||||
|
if err == nil {
|
||||||
|
bp.Name = "unrecovered-panic"
|
||||||
|
bp.ID = -1
|
||||||
|
dbp.breakpointIDCounter--
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return dbp, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) ClearInternalBreakpoints() error {
|
||||||
|
for _, bp := range dbp.breakpoints {
|
||||||
|
if !bp.Internal() {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if _, err := dbp.ClearBreakpoint(bp.Addr); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for i := range dbp.threads {
|
||||||
|
if dbp.threads[i].CurrentBreakpoint != nil && dbp.threads[i].CurrentBreakpoint.Internal() {
|
||||||
|
dbp.threads[i].CurrentBreakpoint = nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) handlePtraceFuncs() {
|
||||||
|
// We must ensure here that we are running on the same thread during
|
||||||
|
// while invoking the ptrace(2) syscall. This is due to the fact that ptrace(2) expects
|
||||||
|
// all commands after PTRACE_ATTACH to come from the same thread.
|
||||||
|
runtime.LockOSThread()
|
||||||
|
|
||||||
|
for fn := range dbp.ptraceChan {
|
||||||
|
fn()
|
||||||
|
dbp.ptraceDoneChan <- nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) execPtraceFunc(fn func()) {
|
||||||
|
dbp.ptraceChan <- fn
|
||||||
|
<-dbp.ptraceDoneChan
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) getGoInformation() (ver GoVersion, isextld bool, err error) {
|
||||||
|
vv, err := dbp.EvalPackageVariable("runtime.buildVersion", LoadConfig{true, 0, 64, 0, 0})
|
||||||
|
if err != nil {
|
||||||
|
err = fmt.Errorf("Could not determine version number: %v\n", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if vv.Unreadable != nil {
|
||||||
|
err = fmt.Errorf("Unreadable version number: %v\n", vv.Unreadable)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
ver, ok := ParseVersionString(constant.StringVal(vv.Value))
|
||||||
|
if !ok {
|
||||||
|
err = fmt.Errorf("Could not parse version number: %v\n", vv.Value)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
rdr := dbp.DwarfReader()
|
||||||
|
rdr.Seek(0)
|
||||||
|
for entry, err := rdr.NextCompileUnit(); entry != nil; entry, err = rdr.NextCompileUnit() {
|
||||||
|
if err != nil {
|
||||||
|
return ver, isextld, err
|
||||||
|
}
|
||||||
|
if prod, ok := entry.Val(dwarf.AttrProducer).(string); ok && (strings.HasPrefix(prod, "GNU AS")) {
|
||||||
|
isextld = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// FindGoroutine returns a G struct representing the goroutine
|
||||||
|
// specified by `gid`.
|
||||||
|
func (dbp *Process) FindGoroutine(gid int) (*G, error) {
|
||||||
|
if gid == -1 {
|
||||||
|
return dbp.selectedGoroutine, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
gs, err := dbp.GoroutinesInfo()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
for i := range gs {
|
||||||
|
if gs[i].ID == gid {
|
||||||
|
return gs[i], nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil, fmt.Errorf("Unknown goroutine %d", gid)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ConvertEvalScope returns a new EvalScope in the context of the
|
||||||
|
// specified goroutine ID and stack frame.
|
||||||
|
func (dbp *Process) ConvertEvalScope(gid, frame int) (*EvalScope, error) {
|
||||||
|
if dbp.exited {
|
||||||
|
return nil, &ProcessExitedError{}
|
||||||
|
}
|
||||||
|
g, err := dbp.FindGoroutine(gid)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if g == nil {
|
||||||
|
return dbp.currentThread.Scope()
|
||||||
|
}
|
||||||
|
|
||||||
|
var out EvalScope
|
||||||
|
|
||||||
|
if g.thread == nil {
|
||||||
|
out.Thread = dbp.currentThread
|
||||||
|
} else {
|
||||||
|
out.Thread = g.thread
|
||||||
|
}
|
||||||
|
|
||||||
|
locs, err := g.Stacktrace(frame)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if frame >= len(locs) {
|
||||||
|
return nil, fmt.Errorf("Frame %d does not exist in goroutine %d", frame, gid)
|
||||||
|
}
|
||||||
|
|
||||||
|
out.PC, out.CFA = locs[frame].Current.PC, locs[frame].CFA
|
||||||
|
|
||||||
|
return &out, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) postExit() {
|
||||||
|
dbp.exited = true
|
||||||
|
close(dbp.ptraceChan)
|
||||||
|
close(dbp.ptraceDoneChan)
|
||||||
|
}
|
231
vendor/github.com/derekparker/delve/pkg/proc/proc_darwin.c
generated
vendored
Normal file
231
vendor/github.com/derekparker/delve/pkg/proc/proc_darwin.c
generated
vendored
Normal file
@ -0,0 +1,231 @@
|
|||||||
|
#include "proc_darwin.h"
|
||||||
|
|
||||||
|
static const unsigned char info_plist[]
|
||||||
|
__attribute__ ((section ("__TEXT,__info_plist"),used)) =
|
||||||
|
"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
|
||||||
|
"<!DOCTYPE plist PUBLIC \"-//Apple Computer//DTD PLIST 1.0//EN\""
|
||||||
|
" \"http://www.apple.com/DTDs/PropertyList-1.0.dtd\">\n"
|
||||||
|
"<plist version=\"1.0\">\n"
|
||||||
|
"<dict>\n"
|
||||||
|
" <key>CFBundleIdentifier</key>\n"
|
||||||
|
" <string>org.dlv</string>\n"
|
||||||
|
" <key>CFBundleName</key>\n"
|
||||||
|
" <string>delve</string>\n"
|
||||||
|
" <key>CFBundleVersion</key>\n"
|
||||||
|
" <string>1.0</string>\n"
|
||||||
|
" <key>SecTaskAccess</key>\n"
|
||||||
|
" <array>\n"
|
||||||
|
" <string>allowed</string>\n"
|
||||||
|
" <string>debug</string>\n"
|
||||||
|
" </array>\n"
|
||||||
|
"</dict>\n"
|
||||||
|
"</plist>\n";
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
acquire_mach_task(int tid,
|
||||||
|
task_t *task,
|
||||||
|
mach_port_t *port_set,
|
||||||
|
mach_port_t *exception_port,
|
||||||
|
mach_port_t *notification_port)
|
||||||
|
{
|
||||||
|
kern_return_t kret;
|
||||||
|
mach_port_t prev_not;
|
||||||
|
mach_port_t self = mach_task_self();
|
||||||
|
|
||||||
|
kret = task_for_pid(self, tid, task);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
// Allocate exception port.
|
||||||
|
kret = mach_port_allocate(self, MACH_PORT_RIGHT_RECEIVE, exception_port);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
kret = mach_port_insert_right(self, *exception_port, *exception_port, MACH_MSG_TYPE_MAKE_SEND);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
kret = task_set_exception_ports(*task, EXC_MASK_BREAKPOINT|EXC_MASK_SOFTWARE, *exception_port,
|
||||||
|
EXCEPTION_DEFAULT, THREAD_STATE_NONE);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
// Allocate notification port to alert of when the process dies.
|
||||||
|
kret = mach_port_allocate(self, MACH_PORT_RIGHT_RECEIVE, notification_port);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
kret = mach_port_insert_right(self, *notification_port, *notification_port, MACH_MSG_TYPE_MAKE_SEND);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
kret = mach_port_request_notification(self, *task, MACH_NOTIFY_DEAD_NAME, 0, *notification_port,
|
||||||
|
MACH_MSG_TYPE_MAKE_SEND_ONCE, &prev_not);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
// Create port set.
|
||||||
|
kret = mach_port_allocate(self, MACH_PORT_RIGHT_PORT_SET, port_set);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
// Move exception and notification ports to port set.
|
||||||
|
kret = mach_port_move_member(self, *exception_port, *port_set);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
return mach_port_move_member(self, *notification_port, *port_set);
|
||||||
|
}
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
reset_exception_ports(task_t task, mach_port_t *exception_port, mach_port_t *notification_port) {
|
||||||
|
kern_return_t kret;
|
||||||
|
mach_port_t prev_not;
|
||||||
|
mach_port_t self = mach_task_self();
|
||||||
|
|
||||||
|
kret = task_set_exception_ports(task, EXC_MASK_BREAKPOINT|EXC_MASK_SOFTWARE, *exception_port,
|
||||||
|
EXCEPTION_DEFAULT, THREAD_STATE_NONE);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
kret = mach_port_request_notification(self, task, MACH_NOTIFY_DEAD_NAME, 0, *notification_port,
|
||||||
|
MACH_MSG_TYPE_MAKE_SEND_ONCE, &prev_not);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
return KERN_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *
|
||||||
|
find_executable(int pid) {
|
||||||
|
static char pathbuf[PATH_MAX];
|
||||||
|
proc_pidpath(pid, pathbuf, PATH_MAX);
|
||||||
|
return pathbuf;
|
||||||
|
}
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
get_threads(task_t task, void *slice, int limit) {
|
||||||
|
kern_return_t kret;
|
||||||
|
thread_act_array_t list;
|
||||||
|
mach_msg_type_number_t count;
|
||||||
|
|
||||||
|
kret = task_threads(task, &list, &count);
|
||||||
|
if (kret != KERN_SUCCESS) {
|
||||||
|
return kret;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (count > limit) {
|
||||||
|
vm_deallocate(mach_task_self(), (vm_address_t) list, count * sizeof(list[0]));
|
||||||
|
return -2;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(slice, (void*)list, count*sizeof(list[0]));
|
||||||
|
|
||||||
|
kret = vm_deallocate(mach_task_self(), (vm_address_t) list, count * sizeof(list[0]));
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
return (kern_return_t)0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
thread_count(task_t task) {
|
||||||
|
kern_return_t kret;
|
||||||
|
thread_act_array_t list;
|
||||||
|
mach_msg_type_number_t count;
|
||||||
|
|
||||||
|
kret = task_threads(task, &list, &count);
|
||||||
|
if (kret != KERN_SUCCESS) return -1;
|
||||||
|
|
||||||
|
kret = vm_deallocate(mach_task_self(), (vm_address_t) list, count * sizeof(list[0]));
|
||||||
|
if (kret != KERN_SUCCESS) return -1;
|
||||||
|
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
mach_port_t
|
||||||
|
mach_port_wait(mach_port_t port_set, task_t *task, int nonblocking) {
|
||||||
|
kern_return_t kret;
|
||||||
|
thread_act_t thread;
|
||||||
|
NDR_record_t *ndr;
|
||||||
|
integer_t *data;
|
||||||
|
union
|
||||||
|
{
|
||||||
|
mach_msg_header_t hdr;
|
||||||
|
char data[256];
|
||||||
|
} msg;
|
||||||
|
mach_msg_option_t opts = MACH_RCV_MSG|MACH_RCV_INTERRUPT;
|
||||||
|
if (nonblocking) {
|
||||||
|
opts |= MACH_RCV_TIMEOUT;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Wait for mach msg.
|
||||||
|
kret = mach_msg(&msg.hdr, opts,
|
||||||
|
0, sizeof(msg.data), port_set, 10, MACH_PORT_NULL);
|
||||||
|
if (kret == MACH_RCV_INTERRUPTED) return kret;
|
||||||
|
if (kret != MACH_MSG_SUCCESS) return 0;
|
||||||
|
|
||||||
|
|
||||||
|
switch (msg.hdr.msgh_id) {
|
||||||
|
case 2401: { // Exception
|
||||||
|
// 2401 is the exception_raise event, defined in:
|
||||||
|
// http://opensource.apple.com/source/xnu/xnu-2422.1.72/osfmk/mach/exc.defs?txt
|
||||||
|
// compile this file with mig to get the C version of the description
|
||||||
|
|
||||||
|
mach_msg_body_t *bod = (mach_msg_body_t*)(&msg.hdr + 1);
|
||||||
|
mach_msg_port_descriptor_t *desc = (mach_msg_port_descriptor_t *)(bod + 1);
|
||||||
|
thread = desc[0].name;
|
||||||
|
*task = desc[1].name;
|
||||||
|
ndr = (NDR_record_t *)(desc + 2);
|
||||||
|
data = (integer_t *)(ndr + 1);
|
||||||
|
|
||||||
|
if (thread_suspend(thread) != KERN_SUCCESS) return 0;
|
||||||
|
// Send our reply back so the kernel knows this exception has been handled.
|
||||||
|
kret = mach_send_reply(msg.hdr);
|
||||||
|
if (kret != MACH_MSG_SUCCESS) return 0;
|
||||||
|
if (data[2] == EXC_SOFT_SIGNAL) {
|
||||||
|
if (data[3] != SIGTRAP) {
|
||||||
|
if (thread_resume(thread) != KERN_SUCCESS) return 0;
|
||||||
|
return mach_port_wait(port_set, task, nonblocking);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return thread;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 72: { // Death
|
||||||
|
// 72 is mach_notify_dead_name, defined in:
|
||||||
|
// https://opensource.apple.com/source/xnu/xnu-1228.7.58/osfmk/mach/notify.defs?txt
|
||||||
|
// compile this file with mig to get the C version of the description
|
||||||
|
ndr = (NDR_record_t *)(&msg.hdr + 1);
|
||||||
|
*task = *((mach_port_name_t *)(ndr + 1));
|
||||||
|
return msg.hdr.msgh_local_port;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
mach_send_reply(mach_msg_header_t hdr) {
|
||||||
|
mig_reply_error_t reply;
|
||||||
|
mach_msg_header_t *rh = &reply.Head;
|
||||||
|
rh->msgh_bits = MACH_MSGH_BITS(MACH_MSGH_BITS_REMOTE(hdr.msgh_bits), 0);
|
||||||
|
rh->msgh_remote_port = hdr.msgh_remote_port;
|
||||||
|
rh->msgh_size = (mach_msg_size_t) sizeof(mig_reply_error_t);
|
||||||
|
rh->msgh_local_port = MACH_PORT_NULL;
|
||||||
|
rh->msgh_id = hdr.msgh_id + 100;
|
||||||
|
|
||||||
|
reply.NDR = NDR_record;
|
||||||
|
reply.RetCode = KERN_SUCCESS;
|
||||||
|
|
||||||
|
return mach_msg(&reply.Head, MACH_SEND_MSG|MACH_SEND_INTERRUPT, rh->msgh_size, 0,
|
||||||
|
MACH_PORT_NULL, MACH_MSG_TIMEOUT_NONE, MACH_PORT_NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
raise_exception(mach_port_t task, mach_port_t thread, mach_port_t exception_port, exception_type_t exception) {
|
||||||
|
return exception_raise(exception_port, thread, task, exception, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
task_t
|
||||||
|
get_task_for_pid(int pid) {
|
||||||
|
task_t task = 0;
|
||||||
|
mach_port_t self = mach_task_self();
|
||||||
|
|
||||||
|
task_for_pid(self, pid, &task);
|
||||||
|
return task;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
task_is_valid(task_t task) {
|
||||||
|
struct task_basic_info info;
|
||||||
|
mach_msg_type_number_t count = TASK_BASIC_INFO_COUNT;
|
||||||
|
return task_info(task, TASK_BASIC_INFO, (task_info_t)&info, &count) == KERN_SUCCESS;
|
||||||
|
}
|
511
vendor/github.com/derekparker/delve/pkg/proc/proc_darwin.go
generated
vendored
Normal file
511
vendor/github.com/derekparker/delve/pkg/proc/proc_darwin.go
generated
vendored
Normal file
@ -0,0 +1,511 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
// #include "proc_darwin.h"
|
||||||
|
// #include "threads_darwin.h"
|
||||||
|
// #include "exec_darwin.h"
|
||||||
|
// #include <stdlib.h>
|
||||||
|
import "C"
|
||||||
|
import (
|
||||||
|
"debug/gosym"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"path/filepath"
|
||||||
|
"sync"
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
"golang.org/x/debug/macho"
|
||||||
|
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/frame"
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/line"
|
||||||
|
sys "golang.org/x/sys/unix"
|
||||||
|
)
|
||||||
|
|
||||||
|
// OSProcessDetails holds Darwin specific information.
|
||||||
|
type OSProcessDetails struct {
|
||||||
|
task C.task_t // mach task for the debugged process.
|
||||||
|
exceptionPort C.mach_port_t // mach port for receiving mach exceptions.
|
||||||
|
notificationPort C.mach_port_t // mach port for dead name notification (process exit).
|
||||||
|
initialized bool
|
||||||
|
|
||||||
|
// the main port we use, will return messages from both the
|
||||||
|
// exception and notification ports.
|
||||||
|
portSet C.mach_port_t
|
||||||
|
}
|
||||||
|
|
||||||
|
// Launch creates and begins debugging a new process. Uses a
|
||||||
|
// custom fork/exec process in order to take advantage of
|
||||||
|
// PT_SIGEXC on Darwin which will turn Unix signals into
|
||||||
|
// Mach exceptions.
|
||||||
|
func Launch(cmd []string, wd string) (*Process, error) {
|
||||||
|
// check that the argument to Launch is an executable file
|
||||||
|
if fi, staterr := os.Stat(cmd[0]); staterr == nil && (fi.Mode()&0111) == 0 {
|
||||||
|
return nil, NotExecutableErr
|
||||||
|
}
|
||||||
|
argv0Go, err := filepath.Abs(cmd[0])
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
// Make sure the binary exists.
|
||||||
|
if filepath.Base(cmd[0]) == cmd[0] {
|
||||||
|
if _, err := exec.LookPath(cmd[0]); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if _, err := os.Stat(argv0Go); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
argv0 := C.CString(argv0Go)
|
||||||
|
argvSlice := make([]*C.char, 0, len(cmd)+1)
|
||||||
|
for _, arg := range cmd {
|
||||||
|
argvSlice = append(argvSlice, C.CString(arg))
|
||||||
|
}
|
||||||
|
// argv array must be null terminated.
|
||||||
|
argvSlice = append(argvSlice, nil)
|
||||||
|
|
||||||
|
dbp := New(0)
|
||||||
|
var pid int
|
||||||
|
dbp.execPtraceFunc(func() {
|
||||||
|
ret := C.fork_exec(argv0, &argvSlice[0], C.int(len(argvSlice)),
|
||||||
|
C.CString(wd),
|
||||||
|
&dbp.os.task, &dbp.os.portSet, &dbp.os.exceptionPort,
|
||||||
|
&dbp.os.notificationPort)
|
||||||
|
pid = int(ret)
|
||||||
|
})
|
||||||
|
if pid <= 0 {
|
||||||
|
return nil, fmt.Errorf("could not fork/exec")
|
||||||
|
}
|
||||||
|
dbp.pid = pid
|
||||||
|
for i := range argvSlice {
|
||||||
|
C.free(unsafe.Pointer(argvSlice[i]))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initialize enough of the Process state so that we can use resume and
|
||||||
|
// trapWait to wait until the child process calls execve.
|
||||||
|
|
||||||
|
for {
|
||||||
|
err = dbp.updateThreadListForTask(C.get_task_for_pid(C.int(dbp.pid)))
|
||||||
|
if err == nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if err != couldNotGetThreadCount && err != couldNotGetThreadList {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := dbp.resume(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
dbp.allGCache = nil
|
||||||
|
for _, th := range dbp.threads {
|
||||||
|
th.clearBreakpointState()
|
||||||
|
}
|
||||||
|
|
||||||
|
trapthread, err := dbp.trapWait(-1)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if err := dbp.Halt(); err != nil {
|
||||||
|
return nil, dbp.exitGuard(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = dbp.waitForStop()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
dbp.os.initialized = true
|
||||||
|
dbp, err = initializeDebugProcess(dbp, argv0Go, false)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := dbp.SwitchThread(trapthread.ID); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return dbp, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Attach to an existing process with the given PID.
|
||||||
|
func Attach(pid int) (*Process, error) {
|
||||||
|
dbp := New(pid)
|
||||||
|
|
||||||
|
kret := C.acquire_mach_task(C.int(pid),
|
||||||
|
&dbp.os.task, &dbp.os.portSet, &dbp.os.exceptionPort,
|
||||||
|
&dbp.os.notificationPort)
|
||||||
|
|
||||||
|
if kret != C.KERN_SUCCESS {
|
||||||
|
return nil, fmt.Errorf("could not attach to %d", pid)
|
||||||
|
}
|
||||||
|
|
||||||
|
dbp.os.initialized = true
|
||||||
|
|
||||||
|
return initializeDebugProcess(dbp, "", true)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Kill kills the process.
|
||||||
|
func (dbp *Process) Kill() (err error) {
|
||||||
|
if dbp.exited {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
err = sys.Kill(-dbp.pid, sys.SIGKILL)
|
||||||
|
if err != nil {
|
||||||
|
return errors.New("could not deliver signal: " + err.Error())
|
||||||
|
}
|
||||||
|
for port := range dbp.threads {
|
||||||
|
if C.thread_resume(C.thread_act_t(port)) != C.KERN_SUCCESS {
|
||||||
|
return errors.New("could not resume task")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for {
|
||||||
|
var task C.task_t
|
||||||
|
port := C.mach_port_wait(dbp.os.portSet, &task, C.int(0))
|
||||||
|
if port == dbp.os.notificationPort {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
dbp.postExit()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) requestManualStop() (err error) {
|
||||||
|
var (
|
||||||
|
task = C.mach_port_t(dbp.os.task)
|
||||||
|
thread = C.mach_port_t(dbp.currentThread.os.threadAct)
|
||||||
|
exceptionPort = C.mach_port_t(dbp.os.exceptionPort)
|
||||||
|
)
|
||||||
|
kret := C.raise_exception(task, thread, exceptionPort, C.EXC_BREAKPOINT)
|
||||||
|
if kret != C.KERN_SUCCESS {
|
||||||
|
return fmt.Errorf("could not raise mach exception")
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var couldNotGetThreadCount = errors.New("could not get thread count")
|
||||||
|
var couldNotGetThreadList = errors.New("could not get thread list")
|
||||||
|
|
||||||
|
func (dbp *Process) updateThreadList() error {
|
||||||
|
return dbp.updateThreadListForTask(dbp.os.task)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) updateThreadListForTask(task C.task_t) error {
|
||||||
|
var (
|
||||||
|
err error
|
||||||
|
kret C.kern_return_t
|
||||||
|
count C.int
|
||||||
|
list []uint32
|
||||||
|
)
|
||||||
|
|
||||||
|
for {
|
||||||
|
count = C.thread_count(task)
|
||||||
|
if count == -1 {
|
||||||
|
return couldNotGetThreadCount
|
||||||
|
}
|
||||||
|
list = make([]uint32, count)
|
||||||
|
|
||||||
|
// TODO(dp) might be better to malloc mem in C and then free it here
|
||||||
|
// instead of getting count above and passing in a slice
|
||||||
|
kret = C.get_threads(task, unsafe.Pointer(&list[0]), count)
|
||||||
|
if kret != -2 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if kret != C.KERN_SUCCESS {
|
||||||
|
return couldNotGetThreadList
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, thread := range dbp.threads {
|
||||||
|
thread.os.exists = false
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, port := range list {
|
||||||
|
thread, ok := dbp.threads[int(port)]
|
||||||
|
if !ok {
|
||||||
|
thread, err = dbp.addThread(int(port), false)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
thread.os.exists = true
|
||||||
|
}
|
||||||
|
|
||||||
|
for threadID, thread := range dbp.threads {
|
||||||
|
if !thread.os.exists {
|
||||||
|
delete(dbp.threads, threadID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) addThread(port int, attach bool) (*Thread, error) {
|
||||||
|
if thread, ok := dbp.threads[port]; ok {
|
||||||
|
return thread, nil
|
||||||
|
}
|
||||||
|
thread := &Thread{
|
||||||
|
ID: port,
|
||||||
|
dbp: dbp,
|
||||||
|
os: new(OSSpecificDetails),
|
||||||
|
}
|
||||||
|
dbp.threads[port] = thread
|
||||||
|
thread.os.threadAct = C.thread_act_t(port)
|
||||||
|
if dbp.currentThread == nil {
|
||||||
|
dbp.SwitchThread(thread.ID)
|
||||||
|
}
|
||||||
|
return thread, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) parseDebugFrame(exe *macho.File, wg *sync.WaitGroup) {
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
|
debugFrameSec := exe.Section("__debug_frame")
|
||||||
|
debugInfoSec := exe.Section("__debug_info")
|
||||||
|
|
||||||
|
if debugFrameSec != nil && debugInfoSec != nil {
|
||||||
|
debugFrame, err := exe.Section("__debug_frame").Data()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get __debug_frame section", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
dat, err := debugInfoSec.Data()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get .debug_info section", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
dbp.frameEntries = frame.Parse(debugFrame, frame.DwarfEndian(dat))
|
||||||
|
} else {
|
||||||
|
fmt.Println("could not find __debug_frame section in binary")
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) obtainGoSymbols(exe *macho.File, wg *sync.WaitGroup) {
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
|
var (
|
||||||
|
symdat []byte
|
||||||
|
pclndat []byte
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
|
||||||
|
if sec := exe.Section("__gosymtab"); sec != nil {
|
||||||
|
symdat, err = sec.Data()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get .gosymtab section", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if sec := exe.Section("__gopclntab"); sec != nil {
|
||||||
|
pclndat, err = sec.Data()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get .gopclntab section", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pcln := gosym.NewLineTable(pclndat, exe.Section("__text").Addr)
|
||||||
|
tab, err := gosym.NewTable(symdat, pcln)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get initialize line table", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
dbp.goSymTable = tab
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) parseDebugLineInfo(exe *macho.File, wg *sync.WaitGroup) {
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
|
if sec := exe.Section("__debug_line"); sec != nil {
|
||||||
|
debugLine, err := exe.Section("__debug_line").Data()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get __debug_line section", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
dbp.lineInfo = line.Parse(debugLine)
|
||||||
|
} else {
|
||||||
|
fmt.Println("could not find __debug_line section in binary")
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var UnsupportedArchErr = errors.New("unsupported architecture - only darwin/amd64 is supported")
|
||||||
|
|
||||||
|
func (dbp *Process) findExecutable(path string) (*macho.File, string, error) {
|
||||||
|
if path == "" {
|
||||||
|
path = C.GoString(C.find_executable(C.int(dbp.pid)))
|
||||||
|
}
|
||||||
|
exe, err := macho.Open(path)
|
||||||
|
if err != nil {
|
||||||
|
return nil, path, err
|
||||||
|
}
|
||||||
|
if exe.Cpu != macho.CpuAmd64 {
|
||||||
|
return nil, path, UnsupportedArchErr
|
||||||
|
}
|
||||||
|
dbp.dwarf, err = exe.DWARF()
|
||||||
|
if err != nil {
|
||||||
|
return nil, path, err
|
||||||
|
}
|
||||||
|
return exe, path, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) trapWait(pid int) (*Thread, error) {
|
||||||
|
for {
|
||||||
|
task := dbp.os.task
|
||||||
|
port := C.mach_port_wait(dbp.os.portSet, &task, C.int(0))
|
||||||
|
|
||||||
|
switch port {
|
||||||
|
case dbp.os.notificationPort:
|
||||||
|
// on macOS >= 10.12.1 the task_t changes after an execve, we could
|
||||||
|
// receive the notification for the death of the pre-execve task_t,
|
||||||
|
// this could also happen *before* we are notified that our task_t has
|
||||||
|
// changed.
|
||||||
|
if dbp.os.task != task {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if !dbp.os.initialized {
|
||||||
|
if pidtask := C.get_task_for_pid(C.int(dbp.pid)); pidtask != 0 && dbp.os.task != pidtask {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_, status, err := dbp.wait(dbp.pid, 0)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
dbp.postExit()
|
||||||
|
return nil, ProcessExitedError{Pid: dbp.pid, Status: status.ExitStatus()}
|
||||||
|
|
||||||
|
case C.MACH_RCV_INTERRUPTED:
|
||||||
|
if !dbp.halt {
|
||||||
|
// Call trapWait again, it seems
|
||||||
|
// MACH_RCV_INTERRUPTED is emitted before
|
||||||
|
// process natural death _sometimes_.
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
return nil, nil
|
||||||
|
|
||||||
|
case 0:
|
||||||
|
return nil, fmt.Errorf("error while waiting for task")
|
||||||
|
}
|
||||||
|
|
||||||
|
// In macOS 10.12.1 if we received a notification for a task other than
|
||||||
|
// the inferior's task and the inferior's task is no longer valid, this
|
||||||
|
// means inferior called execve and its task_t changed.
|
||||||
|
if dbp.os.task != task && C.task_is_valid(dbp.os.task) == 0 {
|
||||||
|
dbp.os.task = task
|
||||||
|
kret := C.reset_exception_ports(dbp.os.task, &dbp.os.exceptionPort, &dbp.os.notificationPort)
|
||||||
|
if kret != C.KERN_SUCCESS {
|
||||||
|
return nil, fmt.Errorf("could not follow task across exec: %d\n", kret)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Since we cannot be notified of new threads on OS X
|
||||||
|
// this is as good a time as any to check for them.
|
||||||
|
dbp.updateThreadList()
|
||||||
|
th, ok := dbp.threads[int(port)]
|
||||||
|
if !ok {
|
||||||
|
if dbp.halt {
|
||||||
|
dbp.halt = false
|
||||||
|
return th, nil
|
||||||
|
}
|
||||||
|
if dbp.firstStart || th.singleStepping {
|
||||||
|
dbp.firstStart = false
|
||||||
|
return th, nil
|
||||||
|
}
|
||||||
|
if err := th.Continue(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
return th, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) waitForStop() ([]int, error) {
|
||||||
|
ports := make([]int, 0, len(dbp.threads))
|
||||||
|
count := 0
|
||||||
|
for {
|
||||||
|
var task C.task_t
|
||||||
|
port := C.mach_port_wait(dbp.os.portSet, &task, C.int(1))
|
||||||
|
if port != 0 && port != dbp.os.notificationPort && port != C.MACH_RCV_INTERRUPTED {
|
||||||
|
count = 0
|
||||||
|
ports = append(ports, int(port))
|
||||||
|
} else {
|
||||||
|
n := C.num_running_threads(dbp.os.task)
|
||||||
|
if n == 0 {
|
||||||
|
return ports, nil
|
||||||
|
} else if n < 0 {
|
||||||
|
return nil, fmt.Errorf("error waiting for thread stop %d", n)
|
||||||
|
} else if count > 16 {
|
||||||
|
return nil, fmt.Errorf("could not stop process %d", n)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) setCurrentBreakpoints(trapthread *Thread) error {
|
||||||
|
ports, err := dbp.waitForStop()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
trapthread.SetCurrentBreakpoint()
|
||||||
|
for _, port := range ports {
|
||||||
|
if th, ok := dbp.threads[port]; ok {
|
||||||
|
err := th.SetCurrentBreakpoint()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) loadProcessInformation(wg *sync.WaitGroup) {
|
||||||
|
wg.Done()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) wait(pid, options int) (int, *sys.WaitStatus, error) {
|
||||||
|
var status sys.WaitStatus
|
||||||
|
wpid, err := sys.Wait4(pid, &status, options, nil)
|
||||||
|
return wpid, &status, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func killProcess(pid int) error {
|
||||||
|
return sys.Kill(pid, sys.SIGINT)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) exitGuard(err error) error {
|
||||||
|
if err != ErrContinueThread {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
_, status, werr := dbp.wait(dbp.pid, sys.WNOHANG)
|
||||||
|
if werr == nil && status.Exited() {
|
||||||
|
dbp.postExit()
|
||||||
|
return ProcessExitedError{Pid: dbp.pid, Status: status.ExitStatus()}
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) resume() error {
|
||||||
|
// all threads stopped over a breakpoint are made to step over it
|
||||||
|
for _, thread := range dbp.threads {
|
||||||
|
if thread.CurrentBreakpoint != nil {
|
||||||
|
if err := thread.StepInstruction(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
thread.CurrentBreakpoint = nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// everything is resumed
|
||||||
|
for _, thread := range dbp.threads {
|
||||||
|
if err := thread.resume(); err != nil {
|
||||||
|
return dbp.exitGuard(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
54
vendor/github.com/derekparker/delve/pkg/proc/proc_darwin.h
generated
vendored
Normal file
54
vendor/github.com/derekparker/delve/pkg/proc/proc_darwin.h
generated
vendored
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
#include <sys/types.h>
|
||||||
|
#include <libproc.h>
|
||||||
|
#include <mach/mach.h>
|
||||||
|
#include <mach/mach_vm.h>
|
||||||
|
#include "mach_exc.h"
|
||||||
|
#include "exc.h"
|
||||||
|
|
||||||
|
#ifdef mig_external
|
||||||
|
mig_external
|
||||||
|
#else
|
||||||
|
extern
|
||||||
|
#endif /* mig_external */
|
||||||
|
boolean_t exc_server(
|
||||||
|
mach_msg_header_t *InHeadP,
|
||||||
|
mach_msg_header_t *OutHeadP);
|
||||||
|
|
||||||
|
#ifdef mig_external
|
||||||
|
mig_external
|
||||||
|
#else
|
||||||
|
extern
|
||||||
|
#endif /* mig_external */
|
||||||
|
boolean_t mach_exc_server(
|
||||||
|
mach_msg_header_t *InHeadP,
|
||||||
|
mach_msg_header_t *OutHeadP);
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
acquire_mach_task(int, task_t*, mach_port_t*, mach_port_t*, mach_port_t*);
|
||||||
|
|
||||||
|
char *
|
||||||
|
find_executable(int pid);
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
get_threads(task_t task, void *data,int limit);
|
||||||
|
|
||||||
|
int
|
||||||
|
thread_count(task_t task);
|
||||||
|
|
||||||
|
mach_port_t
|
||||||
|
mach_port_wait(mach_port_t, task_t*, int);
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
mach_send_reply(mach_msg_header_t);
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
raise_exception(mach_port_t, mach_port_t, mach_port_t, exception_type_t);
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
reset_exception_ports(task_t task, mach_port_t *exception_port, mach_port_t *notification_port);
|
||||||
|
|
||||||
|
task_t
|
||||||
|
get_task_for_pid(int pid);
|
||||||
|
|
||||||
|
int
|
||||||
|
task_is_valid(task_t task);
|
489
vendor/github.com/derekparker/delve/pkg/proc/proc_linux.go
generated
vendored
Normal file
489
vendor/github.com/derekparker/delve/pkg/proc/proc_linux.go
generated
vendored
Normal file
@ -0,0 +1,489 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"debug/gosym"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"path/filepath"
|
||||||
|
"regexp"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"syscall"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
sys "golang.org/x/sys/unix"
|
||||||
|
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/frame"
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/line"
|
||||||
|
"golang.org/x/debug/elf"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Process statuses
|
||||||
|
const (
|
||||||
|
StatusSleeping = 'S'
|
||||||
|
StatusRunning = 'R'
|
||||||
|
StatusTraceStop = 't'
|
||||||
|
StatusZombie = 'Z'
|
||||||
|
|
||||||
|
// Kernel 2.6 has TraceStop as T
|
||||||
|
// TODO(derekparker) Since this means something different based on the
|
||||||
|
// version of the kernel ('T' is job control stop on modern 3.x+ kernels) we
|
||||||
|
// may want to differentiate at some point.
|
||||||
|
StatusTraceStopT = 'T'
|
||||||
|
)
|
||||||
|
|
||||||
|
// OSProcessDetails contains Linux specific
|
||||||
|
// process details.
|
||||||
|
type OSProcessDetails struct {
|
||||||
|
comm string
|
||||||
|
}
|
||||||
|
|
||||||
|
// Launch creates and begins debugging a new process. First entry in
|
||||||
|
// `cmd` is the program to run, and then rest are the arguments
|
||||||
|
// to be supplied to that process. `wd` is working directory of the program.
|
||||||
|
func Launch(cmd []string, wd string) (*Process, error) {
|
||||||
|
var (
|
||||||
|
proc *exec.Cmd
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
// check that the argument to Launch is an executable file
|
||||||
|
if fi, staterr := os.Stat(cmd[0]); staterr == nil && (fi.Mode()&0111) == 0 {
|
||||||
|
return nil, NotExecutableErr
|
||||||
|
}
|
||||||
|
dbp := New(0)
|
||||||
|
dbp.execPtraceFunc(func() {
|
||||||
|
proc = exec.Command(cmd[0])
|
||||||
|
proc.Args = cmd
|
||||||
|
proc.Stdout = os.Stdout
|
||||||
|
proc.Stderr = os.Stderr
|
||||||
|
proc.SysProcAttr = &syscall.SysProcAttr{Ptrace: true, Setpgid: true}
|
||||||
|
if wd != "" {
|
||||||
|
proc.Dir = wd
|
||||||
|
}
|
||||||
|
err = proc.Start()
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
dbp.pid = proc.Process.Pid
|
||||||
|
_, _, err = dbp.wait(proc.Process.Pid, 0)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("waiting for target execve failed: %s", err)
|
||||||
|
}
|
||||||
|
return initializeDebugProcess(dbp, proc.Path, false)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Attach to an existing process with the given PID.
|
||||||
|
func Attach(pid int) (*Process, error) {
|
||||||
|
return initializeDebugProcess(New(pid), "", true)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Kill kills the target process.
|
||||||
|
func (dbp *Process) Kill() (err error) {
|
||||||
|
if dbp.exited {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if !dbp.threads[dbp.pid].Stopped() {
|
||||||
|
return errors.New("process must be stopped in order to kill it")
|
||||||
|
}
|
||||||
|
if err = sys.Kill(-dbp.pid, sys.SIGKILL); err != nil {
|
||||||
|
return errors.New("could not deliver signal " + err.Error())
|
||||||
|
}
|
||||||
|
if _, _, err = dbp.wait(dbp.pid, 0); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
dbp.postExit()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) requestManualStop() (err error) {
|
||||||
|
return sys.Kill(dbp.pid, sys.SIGTRAP)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Attach to a newly created thread, and store that thread in our list of
|
||||||
|
// known threads.
|
||||||
|
func (dbp *Process) addThread(tid int, attach bool) (*Thread, error) {
|
||||||
|
if thread, ok := dbp.threads[tid]; ok {
|
||||||
|
return thread, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var err error
|
||||||
|
if attach {
|
||||||
|
dbp.execPtraceFunc(func() { err = sys.PtraceAttach(tid) })
|
||||||
|
if err != nil && err != sys.EPERM {
|
||||||
|
// Do not return err if err == EPERM,
|
||||||
|
// we may already be tracing this thread due to
|
||||||
|
// PTRACE_O_TRACECLONE. We will surely blow up later
|
||||||
|
// if we truly don't have permissions.
|
||||||
|
return nil, fmt.Errorf("could not attach to new thread %d %s", tid, err)
|
||||||
|
}
|
||||||
|
pid, status, err := dbp.wait(tid, 0)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if status.Exited() {
|
||||||
|
return nil, fmt.Errorf("thread already exited %d", pid)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dbp.execPtraceFunc(func() { err = syscall.PtraceSetOptions(tid, syscall.PTRACE_O_TRACECLONE) })
|
||||||
|
if err == syscall.ESRCH {
|
||||||
|
if _, _, err = dbp.wait(tid, 0); err != nil {
|
||||||
|
return nil, fmt.Errorf("error while waiting after adding thread: %d %s", tid, err)
|
||||||
|
}
|
||||||
|
dbp.execPtraceFunc(func() { err = syscall.PtraceSetOptions(tid, syscall.PTRACE_O_TRACECLONE) })
|
||||||
|
if err == syscall.ESRCH {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("could not set options for new traced thread %d %s", tid, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dbp.threads[tid] = &Thread{
|
||||||
|
ID: tid,
|
||||||
|
dbp: dbp,
|
||||||
|
os: new(OSSpecificDetails),
|
||||||
|
}
|
||||||
|
if dbp.currentThread == nil {
|
||||||
|
dbp.SwitchThread(tid)
|
||||||
|
}
|
||||||
|
return dbp.threads[tid], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) updateThreadList() error {
|
||||||
|
tids, _ := filepath.Glob(fmt.Sprintf("/proc/%d/task/*", dbp.pid))
|
||||||
|
for _, tidpath := range tids {
|
||||||
|
tidstr := filepath.Base(tidpath)
|
||||||
|
tid, err := strconv.Atoi(tidstr)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if _, err := dbp.addThread(tid, tid != dbp.pid); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var UnsupportedArchErr = errors.New("unsupported architecture - only linux/amd64 is supported")
|
||||||
|
|
||||||
|
func (dbp *Process) findExecutable(path string) (*elf.File, string, error) {
|
||||||
|
if path == "" {
|
||||||
|
path = fmt.Sprintf("/proc/%d/exe", dbp.pid)
|
||||||
|
}
|
||||||
|
f, err := os.OpenFile(path, 0, os.ModePerm)
|
||||||
|
if err != nil {
|
||||||
|
return nil, path, err
|
||||||
|
}
|
||||||
|
elfFile, err := elf.NewFile(f)
|
||||||
|
if err != nil {
|
||||||
|
return nil, path, err
|
||||||
|
}
|
||||||
|
if elfFile.Machine != elf.EM_X86_64 {
|
||||||
|
return nil, path, UnsupportedArchErr
|
||||||
|
}
|
||||||
|
dbp.dwarf, err = elfFile.DWARF()
|
||||||
|
if err != nil {
|
||||||
|
return nil, path, err
|
||||||
|
}
|
||||||
|
return elfFile, path, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) parseDebugFrame(exe *elf.File, wg *sync.WaitGroup) {
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
|
debugFrameSec := exe.Section(".debug_frame")
|
||||||
|
debugInfoSec := exe.Section(".debug_info")
|
||||||
|
|
||||||
|
if debugFrameSec != nil && debugInfoSec != nil {
|
||||||
|
debugFrame, err := exe.Section(".debug_frame").Data()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get .debug_frame section", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
dat, err := debugInfoSec.Data()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get .debug_info section", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
dbp.frameEntries = frame.Parse(debugFrame, frame.DwarfEndian(dat))
|
||||||
|
} else {
|
||||||
|
fmt.Println("could not find .debug_frame section in binary")
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) obtainGoSymbols(exe *elf.File, wg *sync.WaitGroup) {
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
|
var (
|
||||||
|
symdat []byte
|
||||||
|
pclndat []byte
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
|
||||||
|
if sec := exe.Section(".gosymtab"); sec != nil {
|
||||||
|
symdat, err = sec.Data()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get .gosymtab section", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if sec := exe.Section(".gopclntab"); sec != nil {
|
||||||
|
pclndat, err = sec.Data()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get .gopclntab section", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pcln := gosym.NewLineTable(pclndat, exe.Section(".text").Addr)
|
||||||
|
tab, err := gosym.NewTable(symdat, pcln)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get initialize line table", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
dbp.goSymTable = tab
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) parseDebugLineInfo(exe *elf.File, wg *sync.WaitGroup) {
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
|
if sec := exe.Section(".debug_line"); sec != nil {
|
||||||
|
debugLine, err := exe.Section(".debug_line").Data()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get .debug_line section", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
dbp.lineInfo = line.Parse(debugLine)
|
||||||
|
} else {
|
||||||
|
fmt.Println("could not find .debug_line section in binary")
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) trapWait(pid int) (*Thread, error) {
|
||||||
|
for {
|
||||||
|
wpid, status, err := dbp.wait(pid, 0)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("wait err %s %d", err, pid)
|
||||||
|
}
|
||||||
|
if wpid == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
th, ok := dbp.threads[wpid]
|
||||||
|
if ok {
|
||||||
|
th.Status = (*WaitStatus)(status)
|
||||||
|
}
|
||||||
|
if status.Exited() {
|
||||||
|
if wpid == dbp.pid {
|
||||||
|
dbp.postExit()
|
||||||
|
return nil, ProcessExitedError{Pid: wpid, Status: status.ExitStatus()}
|
||||||
|
}
|
||||||
|
delete(dbp.threads, wpid)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if status.StopSignal() == sys.SIGTRAP && status.TrapCause() == sys.PTRACE_EVENT_CLONE {
|
||||||
|
// A traced thread has cloned a new thread, grab the pid and
|
||||||
|
// add it to our list of traced threads.
|
||||||
|
var cloned uint
|
||||||
|
dbp.execPtraceFunc(func() { cloned, err = sys.PtraceGetEventMsg(wpid) })
|
||||||
|
if err != nil {
|
||||||
|
if err == sys.ESRCH {
|
||||||
|
// thread died while we were adding it
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
return nil, fmt.Errorf("could not get event message: %s", err)
|
||||||
|
}
|
||||||
|
th, err = dbp.addThread(int(cloned), false)
|
||||||
|
if err != nil {
|
||||||
|
if err == sys.ESRCH {
|
||||||
|
// thread died while we were adding it
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if err = th.Continue(); err != nil {
|
||||||
|
if err == sys.ESRCH {
|
||||||
|
// thread died while we were adding it
|
||||||
|
delete(dbp.threads, th.ID)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
return nil, fmt.Errorf("could not continue new thread %d %s", cloned, err)
|
||||||
|
}
|
||||||
|
if err = dbp.threads[int(wpid)].Continue(); err != nil {
|
||||||
|
if err != sys.ESRCH {
|
||||||
|
return nil, fmt.Errorf("could not continue existing thread %d %s", wpid, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if th == nil {
|
||||||
|
// Sometimes we get an unknown thread, ignore it?
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if status.StopSignal() == sys.SIGTRAP && dbp.halt {
|
||||||
|
th.running = false
|
||||||
|
dbp.halt = false
|
||||||
|
return th, nil
|
||||||
|
}
|
||||||
|
if status.StopSignal() == sys.SIGTRAP {
|
||||||
|
th.running = false
|
||||||
|
return th, nil
|
||||||
|
}
|
||||||
|
if th != nil {
|
||||||
|
// TODO(dp) alert user about unexpected signals here.
|
||||||
|
if err := th.resumeWithSig(int(status.StopSignal())); err != nil {
|
||||||
|
if err == sys.ESRCH {
|
||||||
|
return nil, ProcessExitedError{Pid: dbp.pid}
|
||||||
|
}
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) loadProcessInformation(wg *sync.WaitGroup) {
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
|
comm, err := ioutil.ReadFile(fmt.Sprintf("/proc/%d/comm", dbp.pid))
|
||||||
|
if err == nil {
|
||||||
|
// removes newline character
|
||||||
|
comm = bytes.TrimSuffix(comm, []byte("\n"))
|
||||||
|
}
|
||||||
|
|
||||||
|
if comm == nil || len(comm) <= 0 {
|
||||||
|
stat, err := ioutil.ReadFile(fmt.Sprintf("/proc/%d/stat", dbp.pid))
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("Could not read proc stat: %v\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
expr := fmt.Sprintf("%d\\s*\\((.*)\\)", dbp.pid)
|
||||||
|
rexp, err := regexp.Compile(expr)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Printf("Regexp compile error: %v\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
match := rexp.FindSubmatch(stat)
|
||||||
|
if match == nil {
|
||||||
|
fmt.Printf("No match found using regexp '%s' in /proc/%d/stat\n", expr, dbp.pid)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
comm = match[1]
|
||||||
|
}
|
||||||
|
dbp.os.comm = strings.Replace(string(comm), "%", "%%", -1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func status(pid int, comm string) rune {
|
||||||
|
f, err := os.Open(fmt.Sprintf("/proc/%d/stat", pid))
|
||||||
|
if err != nil {
|
||||||
|
return '\000'
|
||||||
|
}
|
||||||
|
defer f.Close()
|
||||||
|
|
||||||
|
var (
|
||||||
|
p int
|
||||||
|
state rune
|
||||||
|
)
|
||||||
|
|
||||||
|
// The second field of /proc/pid/stat is the name of the task in parenthesis.
|
||||||
|
// The name of the task is the base name of the executable for this process limited to TASK_COMM_LEN characters
|
||||||
|
// Since both parenthesis and spaces can appear inside the name of the task and no escaping happens we need to read the name of the executable first
|
||||||
|
// See: include/linux/sched.c:315 and include/linux/sched.c:1510
|
||||||
|
fmt.Fscanf(f, "%d ("+comm+") %c", &p, &state)
|
||||||
|
return state
|
||||||
|
}
|
||||||
|
|
||||||
|
// waitFast is like wait but does not handle process-exit correctly
|
||||||
|
func (dbp *Process) waitFast(pid int) (int, *sys.WaitStatus, error) {
|
||||||
|
var s sys.WaitStatus
|
||||||
|
wpid, err := sys.Wait4(pid, &s, sys.WALL, nil)
|
||||||
|
return wpid, &s, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) wait(pid, options int) (int, *sys.WaitStatus, error) {
|
||||||
|
var s sys.WaitStatus
|
||||||
|
if (pid != dbp.pid) || (options != 0) {
|
||||||
|
wpid, err := sys.Wait4(pid, &s, sys.WALL|options, nil)
|
||||||
|
return wpid, &s, err
|
||||||
|
}
|
||||||
|
// If we call wait4/waitpid on a thread that is the leader of its group,
|
||||||
|
// with options == 0, while ptracing and the thread leader has exited leaving
|
||||||
|
// zombies of its own then waitpid hangs forever this is apparently intended
|
||||||
|
// behaviour in the linux kernel because it's just so convenient.
|
||||||
|
// Therefore we call wait4 in a loop with WNOHANG, sleeping a while between
|
||||||
|
// calls and exiting when either wait4 succeeds or we find out that the thread
|
||||||
|
// has become a zombie.
|
||||||
|
// References:
|
||||||
|
// https://sourceware.org/bugzilla/show_bug.cgi?id=12702
|
||||||
|
// https://sourceware.org/bugzilla/show_bug.cgi?id=10095
|
||||||
|
// https://sourceware.org/bugzilla/attachment.cgi?id=5685
|
||||||
|
for {
|
||||||
|
wpid, err := sys.Wait4(pid, &s, sys.WNOHANG|sys.WALL|options, nil)
|
||||||
|
if err != nil {
|
||||||
|
return 0, nil, err
|
||||||
|
}
|
||||||
|
if wpid != 0 {
|
||||||
|
return wpid, &s, err
|
||||||
|
}
|
||||||
|
if status(pid, dbp.os.comm) == StatusZombie {
|
||||||
|
return pid, nil, nil
|
||||||
|
}
|
||||||
|
time.Sleep(200 * time.Millisecond)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) setCurrentBreakpoints(trapthread *Thread) error {
|
||||||
|
for _, th := range dbp.threads {
|
||||||
|
if th.CurrentBreakpoint == nil {
|
||||||
|
err := th.SetCurrentBreakpoint()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) exitGuard(err error) error {
|
||||||
|
if err != sys.ESRCH {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if status(dbp.pid, dbp.os.comm) == StatusZombie {
|
||||||
|
_, err := dbp.trapWait(-1)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) resume() error {
|
||||||
|
// all threads stopped over a breakpoint are made to step over it
|
||||||
|
for _, thread := range dbp.threads {
|
||||||
|
if thread.CurrentBreakpoint != nil {
|
||||||
|
if err := thread.StepInstruction(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
thread.CurrentBreakpoint = nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// everything is resumed
|
||||||
|
for _, thread := range dbp.threads {
|
||||||
|
if err := thread.resume(); err != nil && err != sys.ESRCH {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func killProcess(pid int) error {
|
||||||
|
return sys.Kill(pid, sys.SIGINT)
|
||||||
|
}
|
672
vendor/github.com/derekparker/delve/pkg/proc/proc_windows.go
generated
vendored
Normal file
672
vendor/github.com/derekparker/delve/pkg/proc/proc_windows.go
generated
vendored
Normal file
@ -0,0 +1,672 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"debug/gosym"
|
||||||
|
"debug/pe"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"os/exec"
|
||||||
|
"path/filepath"
|
||||||
|
"sync"
|
||||||
|
"syscall"
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
sys "golang.org/x/sys/windows"
|
||||||
|
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/frame"
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/line"
|
||||||
|
"golang.org/x/debug/dwarf"
|
||||||
|
)
|
||||||
|
|
||||||
|
// OSProcessDetails holds Windows specific information.
|
||||||
|
type OSProcessDetails struct {
|
||||||
|
hProcess syscall.Handle
|
||||||
|
breakThread int
|
||||||
|
}
|
||||||
|
|
||||||
|
// Launch creates and begins debugging a new process.
|
||||||
|
func Launch(cmd []string, wd string) (*Process, error) {
|
||||||
|
argv0Go, err := filepath.Abs(cmd[0])
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make sure the binary exists and is an executable file
|
||||||
|
if filepath.Base(cmd[0]) == cmd[0] {
|
||||||
|
if _, err := exec.LookPath(cmd[0]); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
peFile, err := openExecutablePath(argv0Go)
|
||||||
|
if err != nil {
|
||||||
|
return nil, NotExecutableErr
|
||||||
|
}
|
||||||
|
peFile.Close()
|
||||||
|
|
||||||
|
// Duplicate the stdin/stdout/stderr handles
|
||||||
|
files := []uintptr{uintptr(syscall.Stdin), uintptr(syscall.Stdout), uintptr(syscall.Stderr)}
|
||||||
|
p, _ := syscall.GetCurrentProcess()
|
||||||
|
fd := make([]syscall.Handle, len(files))
|
||||||
|
for i := range files {
|
||||||
|
err := syscall.DuplicateHandle(p, syscall.Handle(files[i]), p, &fd[i], 0, true, syscall.DUPLICATE_SAME_ACCESS)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer syscall.CloseHandle(syscall.Handle(fd[i]))
|
||||||
|
}
|
||||||
|
|
||||||
|
argv0, err := syscall.UTF16PtrFromString(argv0Go)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// create suitable command line for CreateProcess
|
||||||
|
// see https://github.com/golang/go/blob/master/src/syscall/exec_windows.go#L326
|
||||||
|
// adapted from standard library makeCmdLine
|
||||||
|
// see https://github.com/golang/go/blob/master/src/syscall/exec_windows.go#L86
|
||||||
|
var cmdLineGo string
|
||||||
|
if len(cmd) >= 1 {
|
||||||
|
for _, v := range cmd {
|
||||||
|
if cmdLineGo != "" {
|
||||||
|
cmdLineGo += " "
|
||||||
|
}
|
||||||
|
cmdLineGo += syscall.EscapeArg(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var cmdLine *uint16
|
||||||
|
if cmdLineGo != "" {
|
||||||
|
if cmdLine, err = syscall.UTF16PtrFromString(cmdLineGo); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var workingDir *uint16
|
||||||
|
if wd != "" {
|
||||||
|
if workingDir, err = syscall.UTF16PtrFromString(wd); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initialize the startup info and create process
|
||||||
|
si := new(sys.StartupInfo)
|
||||||
|
si.Cb = uint32(unsafe.Sizeof(*si))
|
||||||
|
si.Flags = syscall.STARTF_USESTDHANDLES
|
||||||
|
si.StdInput = sys.Handle(fd[0])
|
||||||
|
si.StdOutput = sys.Handle(fd[1])
|
||||||
|
si.StdErr = sys.Handle(fd[2])
|
||||||
|
pi := new(sys.ProcessInformation)
|
||||||
|
|
||||||
|
dbp := New(0)
|
||||||
|
dbp.execPtraceFunc(func() {
|
||||||
|
if wd == "" {
|
||||||
|
err = sys.CreateProcess(argv0, cmdLine, nil, nil, true, _DEBUG_ONLY_THIS_PROCESS, nil, nil, si, pi)
|
||||||
|
} else {
|
||||||
|
err = sys.CreateProcess(argv0, cmdLine, nil, nil, true, _DEBUG_ONLY_THIS_PROCESS, nil, workingDir, si, pi)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
sys.CloseHandle(sys.Handle(pi.Process))
|
||||||
|
sys.CloseHandle(sys.Handle(pi.Thread))
|
||||||
|
|
||||||
|
dbp.pid = int(pi.ProcessId)
|
||||||
|
|
||||||
|
return newDebugProcess(dbp, argv0Go)
|
||||||
|
}
|
||||||
|
|
||||||
|
// newDebugProcess prepares process pid for debugging.
|
||||||
|
func newDebugProcess(dbp *Process, exepath string) (*Process, error) {
|
||||||
|
// It should not actually be possible for the
|
||||||
|
// call to waitForDebugEvent to fail, since Windows
|
||||||
|
// will always fire a CREATE_PROCESS_DEBUG_EVENT event
|
||||||
|
// immediately after launching under DEBUG_ONLY_THIS_PROCESS.
|
||||||
|
// Attaching with DebugActiveProcess has similar effect.
|
||||||
|
var err error
|
||||||
|
var tid, exitCode int
|
||||||
|
dbp.execPtraceFunc(func() {
|
||||||
|
tid, exitCode, err = dbp.waitForDebugEvent(waitBlocking)
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if tid == 0 {
|
||||||
|
dbp.postExit()
|
||||||
|
return nil, ProcessExitedError{Pid: dbp.pid, Status: exitCode}
|
||||||
|
}
|
||||||
|
// Suspend all threads so that the call to _ContinueDebugEvent will
|
||||||
|
// not resume the target.
|
||||||
|
for _, thread := range dbp.threads {
|
||||||
|
_, err := _SuspendThread(thread.os.hThread)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dbp.execPtraceFunc(func() {
|
||||||
|
err = _ContinueDebugEvent(uint32(dbp.pid), uint32(dbp.os.breakThread), _DBG_CONTINUE)
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return initializeDebugProcess(dbp, exepath, false)
|
||||||
|
}
|
||||||
|
|
||||||
|
// findExePath searches for process pid, and returns its executable path.
|
||||||
|
func findExePath(pid int) (string, error) {
|
||||||
|
// Original code suggested different approach (see below).
|
||||||
|
// Maybe it could be useful in the future.
|
||||||
|
//
|
||||||
|
// Find executable path from PID/handle on Windows:
|
||||||
|
// https://msdn.microsoft.com/en-us/library/aa366789(VS.85).aspx
|
||||||
|
|
||||||
|
p, err := syscall.OpenProcess(syscall.PROCESS_QUERY_INFORMATION, false, uint32(pid))
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
defer syscall.CloseHandle(p)
|
||||||
|
|
||||||
|
n := uint32(128)
|
||||||
|
for {
|
||||||
|
buf := make([]uint16, int(n))
|
||||||
|
err = _QueryFullProcessImageName(p, 0, &buf[0], &n)
|
||||||
|
switch err {
|
||||||
|
case syscall.ERROR_INSUFFICIENT_BUFFER:
|
||||||
|
// try bigger buffer
|
||||||
|
n *= 2
|
||||||
|
// but stop if it gets too big
|
||||||
|
if n > 10000 {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
case nil:
|
||||||
|
return syscall.UTF16ToString(buf[:n]), nil
|
||||||
|
default:
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Attach to an existing process with the given PID.
|
||||||
|
func Attach(pid int) (*Process, error) {
|
||||||
|
// TODO: Probably should have SeDebugPrivilege before starting here.
|
||||||
|
err := _DebugActiveProcess(uint32(pid))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
exepath, err := findExePath(pid)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return newDebugProcess(New(pid), exepath)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Kill kills the process.
|
||||||
|
func (dbp *Process) Kill() error {
|
||||||
|
if dbp.exited {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if !dbp.threads[dbp.pid].Stopped() {
|
||||||
|
return errors.New("process must be stopped in order to kill it")
|
||||||
|
}
|
||||||
|
// TODO: Should not have to ignore failures here,
|
||||||
|
// but some tests appear to Kill twice causing
|
||||||
|
// this to fail on second attempt.
|
||||||
|
_ = syscall.TerminateProcess(dbp.os.hProcess, 1)
|
||||||
|
dbp.exited = true
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) requestManualStop() error {
|
||||||
|
return _DebugBreakProcess(dbp.os.hProcess)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) updateThreadList() error {
|
||||||
|
// We ignore this request since threads are being
|
||||||
|
// tracked as they are created/killed in waitForDebugEvent.
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) addThread(hThread syscall.Handle, threadID int, attach, suspendNewThreads bool) (*Thread, error) {
|
||||||
|
if thread, ok := dbp.threads[threadID]; ok {
|
||||||
|
return thread, nil
|
||||||
|
}
|
||||||
|
thread := &Thread{
|
||||||
|
ID: threadID,
|
||||||
|
dbp: dbp,
|
||||||
|
os: new(OSSpecificDetails),
|
||||||
|
}
|
||||||
|
thread.os.hThread = hThread
|
||||||
|
dbp.threads[threadID] = thread
|
||||||
|
if dbp.currentThread == nil {
|
||||||
|
dbp.SwitchThread(thread.ID)
|
||||||
|
}
|
||||||
|
if suspendNewThreads {
|
||||||
|
_, err := _SuspendThread(thread.os.hThread)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return thread, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) parseDebugFrame(exe *pe.File, wg *sync.WaitGroup) {
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
|
debugFrameSec := exe.Section(".debug_frame")
|
||||||
|
debugInfoSec := exe.Section(".debug_info")
|
||||||
|
|
||||||
|
if debugFrameSec != nil && debugInfoSec != nil {
|
||||||
|
debugFrame, err := debugFrameSec.Data()
|
||||||
|
if err != nil && uint32(len(debugFrame)) < debugFrameSec.Size {
|
||||||
|
fmt.Println("could not get .debug_frame section", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
if 0 < debugFrameSec.VirtualSize && debugFrameSec.VirtualSize < debugFrameSec.Size {
|
||||||
|
debugFrame = debugFrame[:debugFrameSec.VirtualSize]
|
||||||
|
}
|
||||||
|
dat, err := debugInfoSec.Data()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get .debug_info section", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
dbp.frameEntries = frame.Parse(debugFrame, frame.DwarfEndian(dat))
|
||||||
|
} else {
|
||||||
|
fmt.Println("could not find .debug_frame section in binary")
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Borrowed from https://golang.org/src/cmd/internal/objfile/pe.go
|
||||||
|
func findPESymbol(f *pe.File, name string) (*pe.Symbol, error) {
|
||||||
|
for _, s := range f.Symbols {
|
||||||
|
if s.Name != name {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if s.SectionNumber <= 0 {
|
||||||
|
return nil, fmt.Errorf("symbol %s: invalid section number %d", name, s.SectionNumber)
|
||||||
|
}
|
||||||
|
if len(f.Sections) < int(s.SectionNumber) {
|
||||||
|
return nil, fmt.Errorf("symbol %s: section number %d is larger than max %d", name, s.SectionNumber, len(f.Sections))
|
||||||
|
}
|
||||||
|
return s, nil
|
||||||
|
}
|
||||||
|
return nil, fmt.Errorf("no %s symbol found", name)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Borrowed from https://golang.org/src/cmd/internal/objfile/pe.go
|
||||||
|
func loadPETable(f *pe.File, sname, ename string) ([]byte, error) {
|
||||||
|
ssym, err := findPESymbol(f, sname)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
esym, err := findPESymbol(f, ename)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if ssym.SectionNumber != esym.SectionNumber {
|
||||||
|
return nil, fmt.Errorf("%s and %s symbols must be in the same section", sname, ename)
|
||||||
|
}
|
||||||
|
sect := f.Sections[ssym.SectionNumber-1]
|
||||||
|
data, err := sect.Data()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return data[ssym.Value:esym.Value], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Borrowed from https://golang.org/src/cmd/internal/objfile/pe.go
|
||||||
|
func pcln(exe *pe.File) (textStart uint64, symtab, pclntab []byte, err error) {
|
||||||
|
var imageBase uint64
|
||||||
|
switch oh := exe.OptionalHeader.(type) {
|
||||||
|
case *pe.OptionalHeader32:
|
||||||
|
imageBase = uint64(oh.ImageBase)
|
||||||
|
case *pe.OptionalHeader64:
|
||||||
|
imageBase = oh.ImageBase
|
||||||
|
default:
|
||||||
|
return 0, nil, nil, fmt.Errorf("pe file format not recognized")
|
||||||
|
}
|
||||||
|
if sect := exe.Section(".text"); sect != nil {
|
||||||
|
textStart = imageBase + uint64(sect.VirtualAddress)
|
||||||
|
}
|
||||||
|
if pclntab, err = loadPETable(exe, "runtime.pclntab", "runtime.epclntab"); err != nil {
|
||||||
|
// We didn't find the symbols, so look for the names used in 1.3 and earlier.
|
||||||
|
// TODO: Remove code looking for the old symbols when we no longer care about 1.3.
|
||||||
|
var err2 error
|
||||||
|
if pclntab, err2 = loadPETable(exe, "pclntab", "epclntab"); err2 != nil {
|
||||||
|
return 0, nil, nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if symtab, err = loadPETable(exe, "runtime.symtab", "runtime.esymtab"); err != nil {
|
||||||
|
// Same as above.
|
||||||
|
var err2 error
|
||||||
|
if symtab, err2 = loadPETable(exe, "symtab", "esymtab"); err2 != nil {
|
||||||
|
return 0, nil, nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return textStart, symtab, pclntab, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) obtainGoSymbols(exe *pe.File, wg *sync.WaitGroup) {
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
|
_, symdat, pclndat, err := pcln(exe)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get Go symbols", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
pcln := gosym.NewLineTable(pclndat, uint64(exe.Section(".text").Offset))
|
||||||
|
tab, err := gosym.NewTable(symdat, pcln)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("could not get initialize line table", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
dbp.goSymTable = tab
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) parseDebugLineInfo(exe *pe.File, wg *sync.WaitGroup) {
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
|
if sec := exe.Section(".debug_line"); sec != nil {
|
||||||
|
debugLine, err := sec.Data()
|
||||||
|
if err != nil && uint32(len(debugLine)) < sec.Size {
|
||||||
|
fmt.Println("could not get .debug_line section", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
if 0 < sec.VirtualSize && sec.VirtualSize < sec.Size {
|
||||||
|
debugLine = debugLine[:sec.VirtualSize]
|
||||||
|
}
|
||||||
|
dbp.lineInfo = line.Parse(debugLine)
|
||||||
|
} else {
|
||||||
|
fmt.Println("could not find .debug_line section in binary")
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var UnsupportedArchErr = errors.New("unsupported architecture of windows/386 - only windows/amd64 is supported")
|
||||||
|
|
||||||
|
func (dbp *Process) findExecutable(path string) (*pe.File, string, error) {
|
||||||
|
peFile, err := openExecutablePath(path)
|
||||||
|
if err != nil {
|
||||||
|
return nil, path, err
|
||||||
|
}
|
||||||
|
if peFile.Machine != pe.IMAGE_FILE_MACHINE_AMD64 {
|
||||||
|
return nil, path, UnsupportedArchErr
|
||||||
|
}
|
||||||
|
dbp.dwarf, err = dwarfFromPE(peFile)
|
||||||
|
if err != nil {
|
||||||
|
return nil, path, err
|
||||||
|
}
|
||||||
|
return peFile, path, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func openExecutablePath(path string) (*pe.File, error) {
|
||||||
|
f, err := os.OpenFile(path, 0, os.ModePerm)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return pe.NewFile(f)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Adapted from src/debug/pe/file.go: pe.(*File).DWARF()
|
||||||
|
func dwarfFromPE(f *pe.File) (*dwarf.Data, error) {
|
||||||
|
// There are many other DWARF sections, but these
|
||||||
|
// are the ones the debug/dwarf package uses.
|
||||||
|
// Don't bother loading others.
|
||||||
|
var names = [...]string{"abbrev", "info", "line", "str"}
|
||||||
|
var dat [len(names)][]byte
|
||||||
|
for i, name := range names {
|
||||||
|
name = ".debug_" + name
|
||||||
|
s := f.Section(name)
|
||||||
|
if s == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
b, err := s.Data()
|
||||||
|
if err != nil && uint32(len(b)) < s.Size {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if 0 < s.VirtualSize && s.VirtualSize < s.Size {
|
||||||
|
b = b[:s.VirtualSize]
|
||||||
|
}
|
||||||
|
dat[i] = b
|
||||||
|
}
|
||||||
|
|
||||||
|
abbrev, info, line, str := dat[0], dat[1], dat[2], dat[3]
|
||||||
|
return dwarf.New(abbrev, nil, nil, info, line, nil, nil, str)
|
||||||
|
}
|
||||||
|
|
||||||
|
type waitForDebugEventFlags int
|
||||||
|
|
||||||
|
const (
|
||||||
|
waitBlocking waitForDebugEventFlags = 1 << iota
|
||||||
|
waitSuspendNewThreads
|
||||||
|
)
|
||||||
|
|
||||||
|
func (dbp *Process) waitForDebugEvent(flags waitForDebugEventFlags) (threadID, exitCode int, err error) {
|
||||||
|
var debugEvent _DEBUG_EVENT
|
||||||
|
shouldExit := false
|
||||||
|
for {
|
||||||
|
continueStatus := uint32(_DBG_CONTINUE)
|
||||||
|
var milliseconds uint32 = 0
|
||||||
|
if flags&waitBlocking != 0 {
|
||||||
|
milliseconds = syscall.INFINITE
|
||||||
|
}
|
||||||
|
// Wait for a debug event...
|
||||||
|
err := _WaitForDebugEvent(&debugEvent, milliseconds)
|
||||||
|
if err != nil {
|
||||||
|
return 0, 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// ... handle each event kind ...
|
||||||
|
unionPtr := unsafe.Pointer(&debugEvent.U[0])
|
||||||
|
switch debugEvent.DebugEventCode {
|
||||||
|
case _CREATE_PROCESS_DEBUG_EVENT:
|
||||||
|
debugInfo := (*_CREATE_PROCESS_DEBUG_INFO)(unionPtr)
|
||||||
|
hFile := debugInfo.File
|
||||||
|
if hFile != 0 && hFile != syscall.InvalidHandle {
|
||||||
|
err = syscall.CloseHandle(hFile)
|
||||||
|
if err != nil {
|
||||||
|
return 0, 0, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
dbp.os.hProcess = debugInfo.Process
|
||||||
|
_, err = dbp.addThread(debugInfo.Thread, int(debugEvent.ThreadId), false, flags&waitSuspendNewThreads != 0)
|
||||||
|
if err != nil {
|
||||||
|
return 0, 0, err
|
||||||
|
}
|
||||||
|
break
|
||||||
|
case _CREATE_THREAD_DEBUG_EVENT:
|
||||||
|
debugInfo := (*_CREATE_THREAD_DEBUG_INFO)(unionPtr)
|
||||||
|
_, err = dbp.addThread(debugInfo.Thread, int(debugEvent.ThreadId), false, flags&waitSuspendNewThreads != 0)
|
||||||
|
if err != nil {
|
||||||
|
return 0, 0, err
|
||||||
|
}
|
||||||
|
break
|
||||||
|
case _EXIT_THREAD_DEBUG_EVENT:
|
||||||
|
delete(dbp.threads, int(debugEvent.ThreadId))
|
||||||
|
break
|
||||||
|
case _OUTPUT_DEBUG_STRING_EVENT:
|
||||||
|
//TODO: Handle debug output strings
|
||||||
|
break
|
||||||
|
case _LOAD_DLL_DEBUG_EVENT:
|
||||||
|
debugInfo := (*_LOAD_DLL_DEBUG_INFO)(unionPtr)
|
||||||
|
hFile := debugInfo.File
|
||||||
|
if hFile != 0 && hFile != syscall.InvalidHandle {
|
||||||
|
err = syscall.CloseHandle(hFile)
|
||||||
|
if err != nil {
|
||||||
|
return 0, 0, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break
|
||||||
|
case _UNLOAD_DLL_DEBUG_EVENT:
|
||||||
|
break
|
||||||
|
case _RIP_EVENT:
|
||||||
|
break
|
||||||
|
case _EXCEPTION_DEBUG_EVENT:
|
||||||
|
exception := (*_EXCEPTION_DEBUG_INFO)(unionPtr)
|
||||||
|
tid := int(debugEvent.ThreadId)
|
||||||
|
|
||||||
|
switch code := exception.ExceptionRecord.ExceptionCode; code {
|
||||||
|
case _EXCEPTION_BREAKPOINT:
|
||||||
|
|
||||||
|
// check if the exception address really is a breakpoint instruction, if
|
||||||
|
// it isn't we already removed that breakpoint and we can't deal with
|
||||||
|
// this exception anymore.
|
||||||
|
atbp := true
|
||||||
|
if thread, found := dbp.threads[tid]; found {
|
||||||
|
if data, err := thread.readMemory(exception.ExceptionRecord.ExceptionAddress, dbp.arch.BreakpointSize()); err == nil {
|
||||||
|
instr := dbp.arch.BreakpointInstruction()
|
||||||
|
for i := range instr {
|
||||||
|
if data[i] != instr[i] {
|
||||||
|
atbp = false
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !atbp {
|
||||||
|
thread.SetPC(uint64(exception.ExceptionRecord.ExceptionAddress))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if atbp {
|
||||||
|
dbp.os.breakThread = tid
|
||||||
|
return tid, 0, nil
|
||||||
|
} else {
|
||||||
|
continueStatus = _DBG_CONTINUE
|
||||||
|
}
|
||||||
|
case _EXCEPTION_SINGLE_STEP:
|
||||||
|
dbp.os.breakThread = tid
|
||||||
|
return tid, 0, nil
|
||||||
|
default:
|
||||||
|
continueStatus = _DBG_EXCEPTION_NOT_HANDLED
|
||||||
|
}
|
||||||
|
case _EXIT_PROCESS_DEBUG_EVENT:
|
||||||
|
debugInfo := (*_EXIT_PROCESS_DEBUG_INFO)(unionPtr)
|
||||||
|
exitCode = int(debugInfo.ExitCode)
|
||||||
|
shouldExit = true
|
||||||
|
default:
|
||||||
|
return 0, 0, fmt.Errorf("unknown debug event code: %d", debugEvent.DebugEventCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
// .. and then continue unless we received an event that indicated we should break into debugger.
|
||||||
|
err = _ContinueDebugEvent(debugEvent.ProcessId, debugEvent.ThreadId, continueStatus)
|
||||||
|
if err != nil {
|
||||||
|
return 0, 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if shouldExit {
|
||||||
|
return 0, exitCode, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) trapWait(pid int) (*Thread, error) {
|
||||||
|
var err error
|
||||||
|
var tid, exitCode int
|
||||||
|
dbp.execPtraceFunc(func() {
|
||||||
|
tid, exitCode, err = dbp.waitForDebugEvent(waitBlocking)
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if tid == 0 {
|
||||||
|
dbp.postExit()
|
||||||
|
return nil, ProcessExitedError{Pid: dbp.pid, Status: exitCode}
|
||||||
|
}
|
||||||
|
th := dbp.threads[tid]
|
||||||
|
return th, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) loadProcessInformation(wg *sync.WaitGroup) {
|
||||||
|
wg.Done()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) wait(pid, options int) (int, *sys.WaitStatus, error) {
|
||||||
|
return 0, nil, fmt.Errorf("not implemented: wait")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) setCurrentBreakpoints(trapthread *Thread) error {
|
||||||
|
// While the debug event that stopped the target was being propagated
|
||||||
|
// other target threads could generate other debug events.
|
||||||
|
// After this function we need to know about all the threads
|
||||||
|
// stopped on a breakpoint. To do that we first suspend all target
|
||||||
|
// threads and then repeatedly call _ContinueDebugEvent followed by
|
||||||
|
// waitForDebugEvent in non-blocking mode.
|
||||||
|
// We need to explicitly call SuspendThread because otherwise the
|
||||||
|
// call to _ContinueDebugEvent will resume execution of some of the
|
||||||
|
// target threads.
|
||||||
|
|
||||||
|
err := trapthread.SetCurrentBreakpoint()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, thread := range dbp.threads {
|
||||||
|
thread.running = false
|
||||||
|
_, err := _SuspendThread(thread.os.hThread)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for {
|
||||||
|
var err error
|
||||||
|
var tid int
|
||||||
|
dbp.execPtraceFunc(func() {
|
||||||
|
err = _ContinueDebugEvent(uint32(dbp.pid), uint32(dbp.os.breakThread), _DBG_CONTINUE)
|
||||||
|
if err == nil {
|
||||||
|
tid, _, _ = dbp.waitForDebugEvent(waitSuspendNewThreads)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if tid == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
err = dbp.threads[tid].SetCurrentBreakpoint()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) exitGuard(err error) error {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) resume() error {
|
||||||
|
for _, thread := range dbp.threads {
|
||||||
|
if thread.CurrentBreakpoint != nil {
|
||||||
|
if err := thread.StepInstruction(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
thread.CurrentBreakpoint = nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, thread := range dbp.threads {
|
||||||
|
thread.running = true
|
||||||
|
_, err := _ResumeThread(thread.os.hThread)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func killProcess(pid int) error {
|
||||||
|
p, err := os.FindProcess(pid)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return p.Kill()
|
||||||
|
}
|
28
vendor/github.com/derekparker/delve/pkg/proc/ptrace_darwin.go
generated
vendored
Normal file
28
vendor/github.com/derekparker/delve/pkg/proc/ptrace_darwin.go
generated
vendored
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import sys "golang.org/x/sys/unix"
|
||||||
|
|
||||||
|
// PtraceAttach executes the sys.PtraceAttach call.
|
||||||
|
func PtraceAttach(pid int) error {
|
||||||
|
return sys.PtraceAttach(pid)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PtraceDetach executes the PT_DETACH ptrace call.
|
||||||
|
func PtraceDetach(tid, sig int) error {
|
||||||
|
return ptrace(sys.PT_DETACH, tid, 1, uintptr(sig))
|
||||||
|
}
|
||||||
|
|
||||||
|
// PtraceCont executes the PTRACE_CONT ptrace call.
|
||||||
|
func PtraceCont(tid, sig int) error {
|
||||||
|
return ptrace(sys.PTRACE_CONT, tid, 1, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PtraceSingleStep returns PT_STEP ptrace call.
|
||||||
|
func PtraceSingleStep(tid int) error {
|
||||||
|
return ptrace(sys.PT_STEP, tid, 1, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func ptrace(request, pid int, addr uintptr, data uintptr) (err error) {
|
||||||
|
_, _, err = sys.Syscall6(sys.SYS_PTRACE, uintptr(request), uintptr(pid), uintptr(addr), uintptr(data), 0, 0)
|
||||||
|
return
|
||||||
|
}
|
96
vendor/github.com/derekparker/delve/pkg/proc/ptrace_linux.go
generated
vendored
Normal file
96
vendor/github.com/derekparker/delve/pkg/proc/ptrace_linux.go
generated
vendored
Normal file
@ -0,0 +1,96 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/binary"
|
||||||
|
"syscall"
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
sys "golang.org/x/sys/unix"
|
||||||
|
)
|
||||||
|
|
||||||
|
// PtraceAttach executes the sys.PtraceAttach call.
|
||||||
|
func PtraceAttach(pid int) error {
|
||||||
|
return sys.PtraceAttach(pid)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PtraceDetach calls ptrace(PTRACE_DETACH).
|
||||||
|
func PtraceDetach(tid, sig int) error {
|
||||||
|
_, _, err := sys.Syscall6(sys.SYS_PTRACE, sys.PTRACE_DETACH, uintptr(tid), 1, uintptr(sig), 0, 0)
|
||||||
|
if err != syscall.Errno(0) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// PtraceCont executes ptrace PTRACE_CONT
|
||||||
|
func PtraceCont(tid, sig int) error {
|
||||||
|
return sys.PtraceCont(tid, sig)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PtraceSingleStep executes ptrace PTRACE_SINGLE_STEP.
|
||||||
|
func PtraceSingleStep(tid int) error {
|
||||||
|
return sys.PtraceSingleStep(tid)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PtracePokeUser execute ptrace PTRACE_POKE_USER.
|
||||||
|
func PtracePokeUser(tid int, off, addr uintptr) error {
|
||||||
|
_, _, err := sys.Syscall6(sys.SYS_PTRACE, sys.PTRACE_POKEUSR, uintptr(tid), uintptr(off), uintptr(addr), 0, 0)
|
||||||
|
if err != syscall.Errno(0) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// PtracePeekUser execute ptrace PTRACE_PEEK_USER.
|
||||||
|
func PtracePeekUser(tid int, off uintptr) (uintptr, error) {
|
||||||
|
var val uintptr
|
||||||
|
_, _, err := syscall.Syscall6(syscall.SYS_PTRACE, syscall.PTRACE_PEEKUSR, uintptr(tid), uintptr(off), uintptr(unsafe.Pointer(&val)), 0, 0)
|
||||||
|
if err != syscall.Errno(0) {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
return val, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// PtraceGetRegset returns floating point registers of the specified thread
|
||||||
|
// using PTRACE.
|
||||||
|
// See amd64_linux_fetch_inferior_registers in gdb/amd64-linux-nat.c.html
|
||||||
|
// and amd64_supply_xsave in gdb/amd64-tdep.c.html
|
||||||
|
// and Section 13.1 (and following) of Intel® 64 and IA-32 Architectures Software Developer’s Manual, Volume 1: Basic Architecture
|
||||||
|
func PtraceGetRegset(tid int) (regset PtraceXsave, err error) {
|
||||||
|
_, _, err = syscall.Syscall6(syscall.SYS_PTRACE, sys.PTRACE_GETFPREGS, uintptr(tid), uintptr(0), uintptr(unsafe.Pointer(®set.PtraceFpRegs)), 0, 0)
|
||||||
|
if err == syscall.Errno(0) {
|
||||||
|
err = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var xstateargs [_X86_XSTATE_MAX_SIZE]byte
|
||||||
|
iov := sys.Iovec{Base: &xstateargs[0], Len: _X86_XSTATE_MAX_SIZE}
|
||||||
|
_, _, err = syscall.Syscall6(syscall.SYS_PTRACE, sys.PTRACE_GETREGSET, uintptr(tid), _NT_X86_XSTATE, uintptr(unsafe.Pointer(&iov)), 0, 0)
|
||||||
|
if err != syscall.Errno(0) {
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
err = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if _XSAVE_HEADER_START+_XSAVE_HEADER_LEN >= iov.Len {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
xsaveheader := xstateargs[_XSAVE_HEADER_START : _XSAVE_HEADER_START+_XSAVE_HEADER_LEN]
|
||||||
|
xstate_bv := binary.LittleEndian.Uint64(xsaveheader[0:8])
|
||||||
|
xcomp_bv := binary.LittleEndian.Uint64(xsaveheader[8:16])
|
||||||
|
|
||||||
|
if xcomp_bv&(1<<63) != 0 {
|
||||||
|
// compact format not supported
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if xstate_bv&(1<<2) == 0 {
|
||||||
|
// AVX state not present
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
avxstate := xstateargs[_XSAVE_EXTENDED_REGION_START:iov.Len]
|
||||||
|
regset.AvxState = true
|
||||||
|
copy(regset.YmmSpace[:], avxstate[:len(regset.YmmSpace)])
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
13
vendor/github.com/derekparker/delve/pkg/proc/ptrace_windows.go
generated
vendored
Normal file
13
vendor/github.com/derekparker/delve/pkg/proc/ptrace_windows.go
generated
vendored
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
)
|
||||||
|
|
||||||
|
func PtraceAttach(pid int) error {
|
||||||
|
return fmt.Errorf("not implemented: PtraceAttach")
|
||||||
|
}
|
||||||
|
|
||||||
|
func PtraceDetach(tid, sig int) error {
|
||||||
|
return _DebugActiveProcessStop(uint32(tid))
|
||||||
|
}
|
231
vendor/github.com/derekparker/delve/pkg/proc/registers.go
generated
vendored
Normal file
231
vendor/github.com/derekparker/delve/pkg/proc/registers.go
generated
vendored
Normal file
@ -0,0 +1,231 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/binary"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"math"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Registers is an interface for a generic register type. The
|
||||||
|
// interface encapsulates the generic values / actions
|
||||||
|
// we need independent of arch. The concrete register types
|
||||||
|
// will be different depending on OS/Arch.
|
||||||
|
type Registers interface {
|
||||||
|
PC() uint64
|
||||||
|
SP() uint64
|
||||||
|
BP() uint64
|
||||||
|
CX() uint64
|
||||||
|
TLS() uint64
|
||||||
|
Get(int) (uint64, error)
|
||||||
|
SetPC(*Thread, uint64) error
|
||||||
|
Slice() []Register
|
||||||
|
}
|
||||||
|
|
||||||
|
type Register struct {
|
||||||
|
Name string
|
||||||
|
Value string
|
||||||
|
}
|
||||||
|
|
||||||
|
func appendWordReg(regs []Register, name string, value uint16) []Register {
|
||||||
|
return append(regs, Register{name, fmt.Sprintf("%#04x", value)})
|
||||||
|
}
|
||||||
|
|
||||||
|
func appendDwordReg(regs []Register, name string, value uint32) []Register {
|
||||||
|
return append(regs, Register{name, fmt.Sprintf("%#08x", value)})
|
||||||
|
}
|
||||||
|
|
||||||
|
func appendQwordReg(regs []Register, name string, value uint64) []Register {
|
||||||
|
return append(regs, Register{name, fmt.Sprintf("%#016x", value)})
|
||||||
|
}
|
||||||
|
|
||||||
|
func appendFlagReg(regs []Register, name string, value uint64, descr flagRegisterDescr, size int) []Register {
|
||||||
|
return append(regs, Register{name, descr.Describe(value, size)})
|
||||||
|
}
|
||||||
|
|
||||||
|
func appendX87Reg(regs []Register, index int, exponent uint16, mantissa uint64) []Register {
|
||||||
|
var f float64
|
||||||
|
fset := false
|
||||||
|
|
||||||
|
const (
|
||||||
|
_SIGNBIT = 1 << 15
|
||||||
|
_EXP_BIAS = (1 << 14) - 1 // 2^(n-1) - 1 = 16383
|
||||||
|
_SPECIALEXP = (1 << 15) - 1 // all bits set
|
||||||
|
_HIGHBIT = 1 << 63
|
||||||
|
_QUIETBIT = 1 << 62
|
||||||
|
)
|
||||||
|
|
||||||
|
sign := 1.0
|
||||||
|
if exponent&_SIGNBIT != 0 {
|
||||||
|
sign = -1.0
|
||||||
|
}
|
||||||
|
exponent &= ^uint16(_SIGNBIT)
|
||||||
|
|
||||||
|
NaN := math.NaN()
|
||||||
|
Inf := math.Inf(+1)
|
||||||
|
|
||||||
|
switch exponent {
|
||||||
|
case 0:
|
||||||
|
switch {
|
||||||
|
case mantissa == 0:
|
||||||
|
f = sign * 0.0
|
||||||
|
fset = true
|
||||||
|
case mantissa&_HIGHBIT != 0:
|
||||||
|
f = NaN
|
||||||
|
fset = true
|
||||||
|
}
|
||||||
|
case _SPECIALEXP:
|
||||||
|
switch {
|
||||||
|
case mantissa&_HIGHBIT == 0:
|
||||||
|
f = sign * Inf
|
||||||
|
fset = true
|
||||||
|
default:
|
||||||
|
f = NaN // signaling NaN
|
||||||
|
fset = true
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
if mantissa&_HIGHBIT == 0 {
|
||||||
|
f = NaN
|
||||||
|
fset = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !fset {
|
||||||
|
significand := float64(mantissa) / (1 << 63)
|
||||||
|
f = sign * math.Ldexp(significand, int(exponent-_EXP_BIAS))
|
||||||
|
}
|
||||||
|
|
||||||
|
return append(regs, Register{fmt.Sprintf("ST(%d)", index), fmt.Sprintf("%#04x%016x\t%g", exponent, mantissa, f)})
|
||||||
|
}
|
||||||
|
|
||||||
|
func appendSSEReg(regs []Register, name string, xmm []byte) []Register {
|
||||||
|
buf := bytes.NewReader(xmm)
|
||||||
|
|
||||||
|
var out bytes.Buffer
|
||||||
|
var vi [16]uint8
|
||||||
|
for i := range vi {
|
||||||
|
binary.Read(buf, binary.LittleEndian, &vi[i])
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Fprintf(&out, "0x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x", vi[15], vi[14], vi[13], vi[12], vi[11], vi[10], vi[9], vi[8], vi[7], vi[6], vi[5], vi[4], vi[3], vi[2], vi[1], vi[0])
|
||||||
|
|
||||||
|
fmt.Fprintf(&out, "\tv2_int={ %02x%02x%02x%02x%02x%02x%02x%02x %02x%02x%02x%02x%02x%02x%02x%02x }", vi[7], vi[6], vi[5], vi[4], vi[3], vi[2], vi[1], vi[0], vi[15], vi[14], vi[13], vi[12], vi[11], vi[10], vi[9], vi[8])
|
||||||
|
|
||||||
|
fmt.Fprintf(&out, "\tv4_int={ %02x%02x%02x%02x %02x%02x%02x%02x %02x%02x%02x%02x %02x%02x%02x%02x }", vi[3], vi[2], vi[1], vi[0], vi[7], vi[6], vi[5], vi[4], vi[11], vi[10], vi[9], vi[8], vi[15], vi[14], vi[13], vi[12])
|
||||||
|
|
||||||
|
fmt.Fprintf(&out, "\tv8_int={ %02x%02x %02x%02x %02x%02x %02x%02x %02x%02x %02x%02x %02x%02x %02x%02x }", vi[1], vi[0], vi[3], vi[2], vi[5], vi[4], vi[7], vi[6], vi[9], vi[8], vi[11], vi[10], vi[13], vi[12], vi[15], vi[14])
|
||||||
|
|
||||||
|
fmt.Fprintf(&out, "\tv16_int={ %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x }", vi[0], vi[1], vi[2], vi[3], vi[4], vi[5], vi[6], vi[7], vi[8], vi[9], vi[10], vi[11], vi[12], vi[13], vi[14], vi[15])
|
||||||
|
|
||||||
|
buf.Seek(0, os.SEEK_SET)
|
||||||
|
var v2 [2]float64
|
||||||
|
for i := range v2 {
|
||||||
|
binary.Read(buf, binary.LittleEndian, &v2[i])
|
||||||
|
}
|
||||||
|
fmt.Fprintf(&out, "\tv2_float={ %g %g }", v2[0], v2[1])
|
||||||
|
|
||||||
|
buf.Seek(0, os.SEEK_SET)
|
||||||
|
var v4 [4]float32
|
||||||
|
for i := range v4 {
|
||||||
|
binary.Read(buf, binary.LittleEndian, &v4[i])
|
||||||
|
}
|
||||||
|
fmt.Fprintf(&out, "\tv4_float={ %g %g %g %g }", v4[0], v4[1], v4[2], v4[3])
|
||||||
|
|
||||||
|
return append(regs, Register{name, out.String()})
|
||||||
|
}
|
||||||
|
|
||||||
|
var UnknownRegisterError = errors.New("unknown register")
|
||||||
|
|
||||||
|
// Registers obtains register values from the debugged process.
|
||||||
|
func (t *Thread) Registers(floatingPoint bool) (Registers, error) {
|
||||||
|
return registers(t, floatingPoint)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PC returns the current PC for this thread.
|
||||||
|
func (t *Thread) PC() (uint64, error) {
|
||||||
|
regs, err := t.Registers(false)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
return regs.PC(), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type flagRegisterDescr []flagDescr
|
||||||
|
type flagDescr struct {
|
||||||
|
name string
|
||||||
|
mask uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
var mxcsrDescription flagRegisterDescr = []flagDescr{
|
||||||
|
{"FZ", 1 << 15},
|
||||||
|
{"RZ/RN", 1<<14 | 1<<13},
|
||||||
|
{"PM", 1 << 12},
|
||||||
|
{"UM", 1 << 11},
|
||||||
|
{"OM", 1 << 10},
|
||||||
|
{"ZM", 1 << 9},
|
||||||
|
{"DM", 1 << 8},
|
||||||
|
{"IM", 1 << 7},
|
||||||
|
{"DAZ", 1 << 6},
|
||||||
|
{"PE", 1 << 5},
|
||||||
|
{"UE", 1 << 4},
|
||||||
|
{"OE", 1 << 3},
|
||||||
|
{"ZE", 1 << 2},
|
||||||
|
{"DE", 1 << 1},
|
||||||
|
{"IE", 1 << 0},
|
||||||
|
}
|
||||||
|
|
||||||
|
var eflagsDescription flagRegisterDescr = []flagDescr{
|
||||||
|
{"CF", 1 << 0},
|
||||||
|
{"", 1 << 1},
|
||||||
|
{"PF", 1 << 2},
|
||||||
|
{"AF", 1 << 4},
|
||||||
|
{"ZF", 1 << 6},
|
||||||
|
{"SF", 1 << 7},
|
||||||
|
{"TF", 1 << 8},
|
||||||
|
{"IF", 1 << 9},
|
||||||
|
{"DF", 1 << 10},
|
||||||
|
{"OF", 1 << 11},
|
||||||
|
{"IOPL", 1<<12 | 1<<13},
|
||||||
|
{"NT", 1 << 14},
|
||||||
|
{"RF", 1 << 16},
|
||||||
|
{"VM", 1 << 17},
|
||||||
|
{"AC", 1 << 18},
|
||||||
|
{"VIF", 1 << 19},
|
||||||
|
{"VIP", 1 << 20},
|
||||||
|
{"ID", 1 << 21},
|
||||||
|
}
|
||||||
|
|
||||||
|
func (descr flagRegisterDescr) Mask() uint64 {
|
||||||
|
var r uint64
|
||||||
|
for _, f := range descr {
|
||||||
|
r = r | f.mask
|
||||||
|
}
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
|
func (descr flagRegisterDescr) Describe(reg uint64, bitsize int) string {
|
||||||
|
var r []string
|
||||||
|
for _, f := range descr {
|
||||||
|
if f.name == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// rbm is f.mask with only the right-most bit set:
|
||||||
|
// 0001 1100 -> 0000 0100
|
||||||
|
rbm := f.mask & -f.mask
|
||||||
|
if rbm == f.mask {
|
||||||
|
if reg&f.mask != 0 {
|
||||||
|
r = append(r, f.name)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
x := (reg & f.mask) >> uint64(math.Log2(float64(rbm)))
|
||||||
|
r = append(r, fmt.Sprintf("%s=%x", f.name, x))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if reg & ^descr.Mask() != 0 {
|
||||||
|
r = append(r, fmt.Sprintf("unknown_flags=%x", reg&^descr.Mask()))
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%#0*x\t[%s]", bitsize/4, reg, strings.Join(r, " "))
|
||||||
|
}
|
367
vendor/github.com/derekparker/delve/pkg/proc/registers_darwin_amd64.go
generated
vendored
Normal file
367
vendor/github.com/derekparker/delve/pkg/proc/registers_darwin_amd64.go
generated
vendored
Normal file
@ -0,0 +1,367 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
// #include "threads_darwin.h"
|
||||||
|
import "C"
|
||||||
|
import (
|
||||||
|
"encoding/binary"
|
||||||
|
"fmt"
|
||||||
|
"rsc.io/x86/x86asm"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Regs represents CPU registers on an AMD64 processor.
|
||||||
|
type Regs struct {
|
||||||
|
rax uint64
|
||||||
|
rbx uint64
|
||||||
|
rcx uint64
|
||||||
|
rdx uint64
|
||||||
|
rdi uint64
|
||||||
|
rsi uint64
|
||||||
|
rbp uint64
|
||||||
|
rsp uint64
|
||||||
|
r8 uint64
|
||||||
|
r9 uint64
|
||||||
|
r10 uint64
|
||||||
|
r11 uint64
|
||||||
|
r12 uint64
|
||||||
|
r13 uint64
|
||||||
|
r14 uint64
|
||||||
|
r15 uint64
|
||||||
|
rip uint64
|
||||||
|
rflags uint64
|
||||||
|
cs uint64
|
||||||
|
fs uint64
|
||||||
|
gs uint64
|
||||||
|
gsBase uint64
|
||||||
|
fpregs []Register
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Regs) Slice() []Register {
|
||||||
|
var regs = []struct {
|
||||||
|
k string
|
||||||
|
v uint64
|
||||||
|
}{
|
||||||
|
{"Rip", r.rip},
|
||||||
|
{"Rsp", r.rsp},
|
||||||
|
{"Rax", r.rax},
|
||||||
|
{"Rbx", r.rbx},
|
||||||
|
{"Rcx", r.rcx},
|
||||||
|
{"Rdx", r.rdx},
|
||||||
|
{"Rdi", r.rdi},
|
||||||
|
{"Rsi", r.rsi},
|
||||||
|
{"Rbp", r.rbp},
|
||||||
|
{"R8", r.r8},
|
||||||
|
{"R9", r.r9},
|
||||||
|
{"R10", r.r10},
|
||||||
|
{"R11", r.r11},
|
||||||
|
{"R12", r.r12},
|
||||||
|
{"R13", r.r13},
|
||||||
|
{"R14", r.r14},
|
||||||
|
{"R15", r.r15},
|
||||||
|
{"Rflags", r.rflags},
|
||||||
|
{"Cs", r.cs},
|
||||||
|
{"Fs", r.fs},
|
||||||
|
{"Gs", r.gs},
|
||||||
|
{"Gs_base", r.gsBase},
|
||||||
|
}
|
||||||
|
out := make([]Register, 0, len(regs)+len(r.fpregs))
|
||||||
|
for _, reg := range regs {
|
||||||
|
if reg.k == "Rflags" {
|
||||||
|
out = appendFlagReg(out, reg.k, reg.v, eflagsDescription, 64)
|
||||||
|
} else {
|
||||||
|
out = appendQwordReg(out, reg.k, reg.v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
out = append(out, r.fpregs...)
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
// PC returns the current program counter
|
||||||
|
// i.e. the RIP CPU register.
|
||||||
|
func (r *Regs) PC() uint64 {
|
||||||
|
return r.rip
|
||||||
|
}
|
||||||
|
|
||||||
|
// SP returns the stack pointer location,
|
||||||
|
// i.e. the RSP register.
|
||||||
|
func (r *Regs) SP() uint64 {
|
||||||
|
return r.rsp
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Regs) BP() uint64 {
|
||||||
|
return r.rbp
|
||||||
|
}
|
||||||
|
|
||||||
|
// CX returns the value of the RCX register.
|
||||||
|
func (r *Regs) CX() uint64 {
|
||||||
|
return r.rcx
|
||||||
|
}
|
||||||
|
|
||||||
|
// TLS returns the value of the register
|
||||||
|
// that contains the location of the thread
|
||||||
|
// local storage segment.
|
||||||
|
func (r *Regs) TLS() uint64 {
|
||||||
|
return r.gsBase
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetPC sets the RIP register to the value specified by `pc`.
|
||||||
|
func (r *Regs) SetPC(thread *Thread, pc uint64) error {
|
||||||
|
kret := C.set_pc(thread.os.threadAct, C.uint64_t(pc))
|
||||||
|
if kret != C.KERN_SUCCESS {
|
||||||
|
return fmt.Errorf("could not set pc")
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Regs) Get(n int) (uint64, error) {
|
||||||
|
reg := x86asm.Reg(n)
|
||||||
|
const (
|
||||||
|
mask8 = 0x000f
|
||||||
|
mask16 = 0x00ff
|
||||||
|
mask32 = 0xffff
|
||||||
|
)
|
||||||
|
|
||||||
|
switch reg {
|
||||||
|
// 8-bit
|
||||||
|
case x86asm.AL:
|
||||||
|
return r.rax & mask8, nil
|
||||||
|
case x86asm.CL:
|
||||||
|
return r.rcx & mask8, nil
|
||||||
|
case x86asm.DL:
|
||||||
|
return r.rdx & mask8, nil
|
||||||
|
case x86asm.BL:
|
||||||
|
return r.rbx & mask8, nil
|
||||||
|
case x86asm.AH:
|
||||||
|
return (r.rax >> 8) & mask8, nil
|
||||||
|
case x86asm.CH:
|
||||||
|
return (r.rcx >> 8) & mask8, nil
|
||||||
|
case x86asm.DH:
|
||||||
|
return (r.rdx >> 8) & mask8, nil
|
||||||
|
case x86asm.BH:
|
||||||
|
return (r.rbx >> 8) & mask8, nil
|
||||||
|
case x86asm.SPB:
|
||||||
|
return r.rsp & mask8, nil
|
||||||
|
case x86asm.BPB:
|
||||||
|
return r.rbp & mask8, nil
|
||||||
|
case x86asm.SIB:
|
||||||
|
return r.rsi & mask8, nil
|
||||||
|
case x86asm.DIB:
|
||||||
|
return r.rdi & mask8, nil
|
||||||
|
case x86asm.R8B:
|
||||||
|
return r.r8 & mask8, nil
|
||||||
|
case x86asm.R9B:
|
||||||
|
return r.r9 & mask8, nil
|
||||||
|
case x86asm.R10B:
|
||||||
|
return r.r10 & mask8, nil
|
||||||
|
case x86asm.R11B:
|
||||||
|
return r.r11 & mask8, nil
|
||||||
|
case x86asm.R12B:
|
||||||
|
return r.r12 & mask8, nil
|
||||||
|
case x86asm.R13B:
|
||||||
|
return r.r13 & mask8, nil
|
||||||
|
case x86asm.R14B:
|
||||||
|
return r.r14 & mask8, nil
|
||||||
|
case x86asm.R15B:
|
||||||
|
return r.r15 & mask8, nil
|
||||||
|
|
||||||
|
// 16-bit
|
||||||
|
case x86asm.AX:
|
||||||
|
return r.rax & mask16, nil
|
||||||
|
case x86asm.CX:
|
||||||
|
return r.rcx & mask16, nil
|
||||||
|
case x86asm.DX:
|
||||||
|
return r.rdx & mask16, nil
|
||||||
|
case x86asm.BX:
|
||||||
|
return r.rbx & mask16, nil
|
||||||
|
case x86asm.SP:
|
||||||
|
return r.rsp & mask16, nil
|
||||||
|
case x86asm.BP:
|
||||||
|
return r.rbp & mask16, nil
|
||||||
|
case x86asm.SI:
|
||||||
|
return r.rsi & mask16, nil
|
||||||
|
case x86asm.DI:
|
||||||
|
return r.rdi & mask16, nil
|
||||||
|
case x86asm.R8W:
|
||||||
|
return r.r8 & mask16, nil
|
||||||
|
case x86asm.R9W:
|
||||||
|
return r.r9 & mask16, nil
|
||||||
|
case x86asm.R10W:
|
||||||
|
return r.r10 & mask16, nil
|
||||||
|
case x86asm.R11W:
|
||||||
|
return r.r11 & mask16, nil
|
||||||
|
case x86asm.R12W:
|
||||||
|
return r.r12 & mask16, nil
|
||||||
|
case x86asm.R13W:
|
||||||
|
return r.r13 & mask16, nil
|
||||||
|
case x86asm.R14W:
|
||||||
|
return r.r14 & mask16, nil
|
||||||
|
case x86asm.R15W:
|
||||||
|
return r.r15 & mask16, nil
|
||||||
|
|
||||||
|
// 32-bit
|
||||||
|
case x86asm.EAX:
|
||||||
|
return r.rax & mask32, nil
|
||||||
|
case x86asm.ECX:
|
||||||
|
return r.rcx & mask32, nil
|
||||||
|
case x86asm.EDX:
|
||||||
|
return r.rdx & mask32, nil
|
||||||
|
case x86asm.EBX:
|
||||||
|
return r.rbx & mask32, nil
|
||||||
|
case x86asm.ESP:
|
||||||
|
return r.rsp & mask32, nil
|
||||||
|
case x86asm.EBP:
|
||||||
|
return r.rbp & mask32, nil
|
||||||
|
case x86asm.ESI:
|
||||||
|
return r.rsi & mask32, nil
|
||||||
|
case x86asm.EDI:
|
||||||
|
return r.rdi & mask32, nil
|
||||||
|
case x86asm.R8L:
|
||||||
|
return r.r8 & mask32, nil
|
||||||
|
case x86asm.R9L:
|
||||||
|
return r.r9 & mask32, nil
|
||||||
|
case x86asm.R10L:
|
||||||
|
return r.r10 & mask32, nil
|
||||||
|
case x86asm.R11L:
|
||||||
|
return r.r11 & mask32, nil
|
||||||
|
case x86asm.R12L:
|
||||||
|
return r.r12 & mask32, nil
|
||||||
|
case x86asm.R13L:
|
||||||
|
return r.r13 & mask32, nil
|
||||||
|
case x86asm.R14L:
|
||||||
|
return r.r14 & mask32, nil
|
||||||
|
case x86asm.R15L:
|
||||||
|
return r.r15 & mask32, nil
|
||||||
|
|
||||||
|
// 64-bit
|
||||||
|
case x86asm.RAX:
|
||||||
|
return r.rax, nil
|
||||||
|
case x86asm.RCX:
|
||||||
|
return r.rcx, nil
|
||||||
|
case x86asm.RDX:
|
||||||
|
return r.rdx, nil
|
||||||
|
case x86asm.RBX:
|
||||||
|
return r.rbx, nil
|
||||||
|
case x86asm.RSP:
|
||||||
|
return r.rsp, nil
|
||||||
|
case x86asm.RBP:
|
||||||
|
return r.rbp, nil
|
||||||
|
case x86asm.RSI:
|
||||||
|
return r.rsi, nil
|
||||||
|
case x86asm.RDI:
|
||||||
|
return r.rdi, nil
|
||||||
|
case x86asm.R8:
|
||||||
|
return r.r8, nil
|
||||||
|
case x86asm.R9:
|
||||||
|
return r.r9, nil
|
||||||
|
case x86asm.R10:
|
||||||
|
return r.r10, nil
|
||||||
|
case x86asm.R11:
|
||||||
|
return r.r11, nil
|
||||||
|
case x86asm.R12:
|
||||||
|
return r.r12, nil
|
||||||
|
case x86asm.R13:
|
||||||
|
return r.r13, nil
|
||||||
|
case x86asm.R14:
|
||||||
|
return r.r14, nil
|
||||||
|
case x86asm.R15:
|
||||||
|
return r.r15, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0, UnknownRegisterError
|
||||||
|
}
|
||||||
|
|
||||||
|
func registers(thread *Thread, floatingPoint bool) (Registers, error) {
|
||||||
|
var state C.x86_thread_state64_t
|
||||||
|
var identity C.thread_identifier_info_data_t
|
||||||
|
kret := C.get_registers(C.mach_port_name_t(thread.os.threadAct), &state)
|
||||||
|
if kret != C.KERN_SUCCESS {
|
||||||
|
return nil, fmt.Errorf("could not get registers")
|
||||||
|
}
|
||||||
|
kret = C.get_identity(C.mach_port_name_t(thread.os.threadAct), &identity)
|
||||||
|
if kret != C.KERN_SUCCESS {
|
||||||
|
return nil, fmt.Errorf("could not get thread identity informations")
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
thread_identifier_info::thread_handle contains the base of the
|
||||||
|
thread-specific data area, which on x86 and x86_64 is the thread’s base
|
||||||
|
address of the %gs segment. 10.9.2 xnu-2422.90.20/osfmk/kern/thread.c
|
||||||
|
thread_info_internal() gets the value from
|
||||||
|
machine_thread::cthread_self, which is the same value used to set the
|
||||||
|
%gs base in xnu-2422.90.20/osfmk/i386/pcb_native.c
|
||||||
|
act_machine_switch_pcb().
|
||||||
|
--
|
||||||
|
comment copied from chromium's crashpad
|
||||||
|
https://chromium.googlesource.com/crashpad/crashpad/+/master/snapshot/mac/process_reader.cc
|
||||||
|
*/
|
||||||
|
regs := &Regs{
|
||||||
|
rax: uint64(state.__rax),
|
||||||
|
rbx: uint64(state.__rbx),
|
||||||
|
rcx: uint64(state.__rcx),
|
||||||
|
rdx: uint64(state.__rdx),
|
||||||
|
rdi: uint64(state.__rdi),
|
||||||
|
rsi: uint64(state.__rsi),
|
||||||
|
rbp: uint64(state.__rbp),
|
||||||
|
rsp: uint64(state.__rsp),
|
||||||
|
r8: uint64(state.__r8),
|
||||||
|
r9: uint64(state.__r9),
|
||||||
|
r10: uint64(state.__r10),
|
||||||
|
r11: uint64(state.__r11),
|
||||||
|
r12: uint64(state.__r12),
|
||||||
|
r13: uint64(state.__r13),
|
||||||
|
r14: uint64(state.__r14),
|
||||||
|
r15: uint64(state.__r15),
|
||||||
|
rip: uint64(state.__rip),
|
||||||
|
rflags: uint64(state.__rflags),
|
||||||
|
cs: uint64(state.__cs),
|
||||||
|
fs: uint64(state.__fs),
|
||||||
|
gs: uint64(state.__gs),
|
||||||
|
gsBase: uint64(identity.thread_handle),
|
||||||
|
}
|
||||||
|
|
||||||
|
if floatingPoint {
|
||||||
|
// https://opensource.apple.com/source/xnu/xnu-792.13.8/osfmk/mach/i386/thread_status.h?txt
|
||||||
|
var fpstate C.x86_float_state64_t
|
||||||
|
kret = C.get_fpu_registers(C.mach_port_name_t(thread.os.threadAct), &fpstate)
|
||||||
|
if kret != C.KERN_SUCCESS {
|
||||||
|
return nil, fmt.Errorf("could not get floating point registers")
|
||||||
|
}
|
||||||
|
|
||||||
|
regs.fpregs = appendWordReg(regs.fpregs, "CW", *((*uint16)(unsafe.Pointer(&fpstate.__fpu_fcw))))
|
||||||
|
regs.fpregs = appendWordReg(regs.fpregs, "SW", *((*uint16)(unsafe.Pointer(&fpstate.__fpu_fsw))))
|
||||||
|
regs.fpregs = appendWordReg(regs.fpregs, "TW", uint16(fpstate.__fpu_ftw))
|
||||||
|
regs.fpregs = appendWordReg(regs.fpregs, "FOP", uint16(fpstate.__fpu_fop))
|
||||||
|
regs.fpregs = appendQwordReg(regs.fpregs, "FIP", uint64(fpstate.__fpu_cs)<<32|uint64(fpstate.__fpu_ip))
|
||||||
|
regs.fpregs = appendQwordReg(regs.fpregs, "FDP", uint64(fpstate.__fpu_ds)<<32|uint64(fpstate.__fpu_dp))
|
||||||
|
|
||||||
|
for i, st := range []*C.char{&fpstate.__fpu_stmm0.__mmst_reg[0], &fpstate.__fpu_stmm1.__mmst_reg[0], &fpstate.__fpu_stmm2.__mmst_reg[0], &fpstate.__fpu_stmm3.__mmst_reg[0], &fpstate.__fpu_stmm4.__mmst_reg[0], &fpstate.__fpu_stmm5.__mmst_reg[0], &fpstate.__fpu_stmm6.__mmst_reg[0], &fpstate.__fpu_stmm7.__mmst_reg[0]} {
|
||||||
|
stb := C.GoBytes(unsafe.Pointer(st), 10)
|
||||||
|
mantissa := binary.LittleEndian.Uint64(stb[:8])
|
||||||
|
exponent := binary.LittleEndian.Uint16(stb[8:])
|
||||||
|
regs.fpregs = appendX87Reg(regs.fpregs, i, exponent, mantissa)
|
||||||
|
}
|
||||||
|
|
||||||
|
regs.fpregs = appendFlagReg(regs.fpregs, "MXCSR", uint64(fpstate.__fpu_mxcsr), mxcsrDescription, 32)
|
||||||
|
regs.fpregs = appendDwordReg(regs.fpregs, "MXCSR_MASK", uint32(fpstate.__fpu_mxcsrmask))
|
||||||
|
|
||||||
|
for i, xmm := range []*C.char{&fpstate.__fpu_xmm0.__xmm_reg[0], &fpstate.__fpu_xmm1.__xmm_reg[0], &fpstate.__fpu_xmm2.__xmm_reg[0], &fpstate.__fpu_xmm3.__xmm_reg[0], &fpstate.__fpu_xmm4.__xmm_reg[0], &fpstate.__fpu_xmm5.__xmm_reg[0], &fpstate.__fpu_xmm6.__xmm_reg[0], &fpstate.__fpu_xmm7.__xmm_reg[0], &fpstate.__fpu_xmm8.__xmm_reg[0], &fpstate.__fpu_xmm9.__xmm_reg[0], &fpstate.__fpu_xmm10.__xmm_reg[0], &fpstate.__fpu_xmm11.__xmm_reg[0], &fpstate.__fpu_xmm12.__xmm_reg[0], &fpstate.__fpu_xmm13.__xmm_reg[0], &fpstate.__fpu_xmm14.__xmm_reg[0], &fpstate.__fpu_xmm15.__xmm_reg[0]} {
|
||||||
|
regs.fpregs = appendSSEReg(regs.fpregs, fmt.Sprintf("XMM%d", i), C.GoBytes(unsafe.Pointer(xmm), 16))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return regs, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (thread *Thread) saveRegisters() (Registers, error) {
|
||||||
|
kret := C.get_registers(C.mach_port_name_t(thread.os.threadAct), &thread.os.registers)
|
||||||
|
if kret != C.KERN_SUCCESS {
|
||||||
|
return nil, fmt.Errorf("could not save register contents")
|
||||||
|
}
|
||||||
|
return &Regs{rip: uint64(thread.os.registers.__rip), rsp: uint64(thread.os.registers.__rsp)}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (thread *Thread) restoreRegisters() error {
|
||||||
|
kret := C.set_registers(C.mach_port_name_t(thread.os.threadAct), &thread.os.registers)
|
||||||
|
if kret != C.KERN_SUCCESS {
|
||||||
|
return fmt.Errorf("could not save register contents")
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
329
vendor/github.com/derekparker/delve/pkg/proc/registers_linux_amd64.go
generated
vendored
Normal file
329
vendor/github.com/derekparker/delve/pkg/proc/registers_linux_amd64.go
generated
vendored
Normal file
@ -0,0 +1,329 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"rsc.io/x86/x86asm"
|
||||||
|
|
||||||
|
sys "golang.org/x/sys/unix"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Regs is a wrapper for sys.PtraceRegs.
|
||||||
|
type Regs struct {
|
||||||
|
regs *sys.PtraceRegs
|
||||||
|
fpregs []Register
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Regs) Slice() []Register {
|
||||||
|
var regs = []struct {
|
||||||
|
k string
|
||||||
|
v uint64
|
||||||
|
}{
|
||||||
|
{"Rip", r.regs.Rip},
|
||||||
|
{"Rsp", r.regs.Rsp},
|
||||||
|
{"Rax", r.regs.Rax},
|
||||||
|
{"Rbx", r.regs.Rbx},
|
||||||
|
{"Rcx", r.regs.Rcx},
|
||||||
|
{"Rdx", r.regs.Rdx},
|
||||||
|
{"Rdi", r.regs.Rdi},
|
||||||
|
{"Rsi", r.regs.Rsi},
|
||||||
|
{"Rbp", r.regs.Rbp},
|
||||||
|
{"R8", r.regs.R8},
|
||||||
|
{"R9", r.regs.R9},
|
||||||
|
{"R10", r.regs.R10},
|
||||||
|
{"R11", r.regs.R11},
|
||||||
|
{"R12", r.regs.R12},
|
||||||
|
{"R13", r.regs.R13},
|
||||||
|
{"R14", r.regs.R14},
|
||||||
|
{"R15", r.regs.R15},
|
||||||
|
{"Orig_rax", r.regs.Orig_rax},
|
||||||
|
{"Cs", r.regs.Cs},
|
||||||
|
{"Eflags", r.regs.Eflags},
|
||||||
|
{"Ss", r.regs.Ss},
|
||||||
|
{"Fs_base", r.regs.Fs_base},
|
||||||
|
{"Gs_base", r.regs.Gs_base},
|
||||||
|
{"Ds", r.regs.Ds},
|
||||||
|
{"Es", r.regs.Es},
|
||||||
|
{"Fs", r.regs.Fs},
|
||||||
|
{"Gs", r.regs.Gs},
|
||||||
|
}
|
||||||
|
out := make([]Register, 0, len(regs)+len(r.fpregs))
|
||||||
|
for _, reg := range regs {
|
||||||
|
if reg.k == "Eflags" {
|
||||||
|
out = appendFlagReg(out, reg.k, reg.v, eflagsDescription, 64)
|
||||||
|
} else {
|
||||||
|
out = appendQwordReg(out, reg.k, reg.v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
out = append(out, r.fpregs...)
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
// PC returns the value of RIP register.
|
||||||
|
func (r *Regs) PC() uint64 {
|
||||||
|
return r.regs.PC()
|
||||||
|
}
|
||||||
|
|
||||||
|
// SP returns the value of RSP register.
|
||||||
|
func (r *Regs) SP() uint64 {
|
||||||
|
return r.regs.Rsp
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Regs) BP() uint64 {
|
||||||
|
return r.regs.Rbp
|
||||||
|
}
|
||||||
|
|
||||||
|
// CX returns the value of RCX register.
|
||||||
|
func (r *Regs) CX() uint64 {
|
||||||
|
return r.regs.Rcx
|
||||||
|
}
|
||||||
|
|
||||||
|
// TLS returns the address of the thread
|
||||||
|
// local storage memory segment.
|
||||||
|
func (r *Regs) TLS() uint64 {
|
||||||
|
return r.regs.Fs_base
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetPC sets RIP to the value specified by 'pc'.
|
||||||
|
func (r *Regs) SetPC(thread *Thread, pc uint64) (err error) {
|
||||||
|
r.regs.SetPC(pc)
|
||||||
|
thread.dbp.execPtraceFunc(func() { err = sys.PtraceSetRegs(thread.ID, r.regs) })
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Regs) Get(n int) (uint64, error) {
|
||||||
|
reg := x86asm.Reg(n)
|
||||||
|
const (
|
||||||
|
mask8 = 0x000f
|
||||||
|
mask16 = 0x00ff
|
||||||
|
mask32 = 0xffff
|
||||||
|
)
|
||||||
|
|
||||||
|
switch reg {
|
||||||
|
// 8-bit
|
||||||
|
case x86asm.AL:
|
||||||
|
return r.regs.Rax & mask8, nil
|
||||||
|
case x86asm.CL:
|
||||||
|
return r.regs.Rcx & mask8, nil
|
||||||
|
case x86asm.DL:
|
||||||
|
return r.regs.Rdx & mask8, nil
|
||||||
|
case x86asm.BL:
|
||||||
|
return r.regs.Rbx & mask8, nil
|
||||||
|
case x86asm.AH:
|
||||||
|
return (r.regs.Rax >> 8) & mask8, nil
|
||||||
|
case x86asm.CH:
|
||||||
|
return (r.regs.Rcx >> 8) & mask8, nil
|
||||||
|
case x86asm.DH:
|
||||||
|
return (r.regs.Rdx >> 8) & mask8, nil
|
||||||
|
case x86asm.BH:
|
||||||
|
return (r.regs.Rbx >> 8) & mask8, nil
|
||||||
|
case x86asm.SPB:
|
||||||
|
return r.regs.Rsp & mask8, nil
|
||||||
|
case x86asm.BPB:
|
||||||
|
return r.regs.Rbp & mask8, nil
|
||||||
|
case x86asm.SIB:
|
||||||
|
return r.regs.Rsi & mask8, nil
|
||||||
|
case x86asm.DIB:
|
||||||
|
return r.regs.Rdi & mask8, nil
|
||||||
|
case x86asm.R8B:
|
||||||
|
return r.regs.R8 & mask8, nil
|
||||||
|
case x86asm.R9B:
|
||||||
|
return r.regs.R9 & mask8, nil
|
||||||
|
case x86asm.R10B:
|
||||||
|
return r.regs.R10 & mask8, nil
|
||||||
|
case x86asm.R11B:
|
||||||
|
return r.regs.R11 & mask8, nil
|
||||||
|
case x86asm.R12B:
|
||||||
|
return r.regs.R12 & mask8, nil
|
||||||
|
case x86asm.R13B:
|
||||||
|
return r.regs.R13 & mask8, nil
|
||||||
|
case x86asm.R14B:
|
||||||
|
return r.regs.R14 & mask8, nil
|
||||||
|
case x86asm.R15B:
|
||||||
|
return r.regs.R15 & mask8, nil
|
||||||
|
|
||||||
|
// 16-bit
|
||||||
|
case x86asm.AX:
|
||||||
|
return r.regs.Rax & mask16, nil
|
||||||
|
case x86asm.CX:
|
||||||
|
return r.regs.Rcx & mask16, nil
|
||||||
|
case x86asm.DX:
|
||||||
|
return r.regs.Rdx & mask16, nil
|
||||||
|
case x86asm.BX:
|
||||||
|
return r.regs.Rbx & mask16, nil
|
||||||
|
case x86asm.SP:
|
||||||
|
return r.regs.Rsp & mask16, nil
|
||||||
|
case x86asm.BP:
|
||||||
|
return r.regs.Rbp & mask16, nil
|
||||||
|
case x86asm.SI:
|
||||||
|
return r.regs.Rsi & mask16, nil
|
||||||
|
case x86asm.DI:
|
||||||
|
return r.regs.Rdi & mask16, nil
|
||||||
|
case x86asm.R8W:
|
||||||
|
return r.regs.R8 & mask16, nil
|
||||||
|
case x86asm.R9W:
|
||||||
|
return r.regs.R9 & mask16, nil
|
||||||
|
case x86asm.R10W:
|
||||||
|
return r.regs.R10 & mask16, nil
|
||||||
|
case x86asm.R11W:
|
||||||
|
return r.regs.R11 & mask16, nil
|
||||||
|
case x86asm.R12W:
|
||||||
|
return r.regs.R12 & mask16, nil
|
||||||
|
case x86asm.R13W:
|
||||||
|
return r.regs.R13 & mask16, nil
|
||||||
|
case x86asm.R14W:
|
||||||
|
return r.regs.R14 & mask16, nil
|
||||||
|
case x86asm.R15W:
|
||||||
|
return r.regs.R15 & mask16, nil
|
||||||
|
|
||||||
|
// 32-bit
|
||||||
|
case x86asm.EAX:
|
||||||
|
return r.regs.Rax & mask32, nil
|
||||||
|
case x86asm.ECX:
|
||||||
|
return r.regs.Rcx & mask32, nil
|
||||||
|
case x86asm.EDX:
|
||||||
|
return r.regs.Rdx & mask32, nil
|
||||||
|
case x86asm.EBX:
|
||||||
|
return r.regs.Rbx & mask32, nil
|
||||||
|
case x86asm.ESP:
|
||||||
|
return r.regs.Rsp & mask32, nil
|
||||||
|
case x86asm.EBP:
|
||||||
|
return r.regs.Rbp & mask32, nil
|
||||||
|
case x86asm.ESI:
|
||||||
|
return r.regs.Rsi & mask32, nil
|
||||||
|
case x86asm.EDI:
|
||||||
|
return r.regs.Rdi & mask32, nil
|
||||||
|
case x86asm.R8L:
|
||||||
|
return r.regs.R8 & mask32, nil
|
||||||
|
case x86asm.R9L:
|
||||||
|
return r.regs.R9 & mask32, nil
|
||||||
|
case x86asm.R10L:
|
||||||
|
return r.regs.R10 & mask32, nil
|
||||||
|
case x86asm.R11L:
|
||||||
|
return r.regs.R11 & mask32, nil
|
||||||
|
case x86asm.R12L:
|
||||||
|
return r.regs.R12 & mask32, nil
|
||||||
|
case x86asm.R13L:
|
||||||
|
return r.regs.R13 & mask32, nil
|
||||||
|
case x86asm.R14L:
|
||||||
|
return r.regs.R14 & mask32, nil
|
||||||
|
case x86asm.R15L:
|
||||||
|
return r.regs.R15 & mask32, nil
|
||||||
|
|
||||||
|
// 64-bit
|
||||||
|
case x86asm.RAX:
|
||||||
|
return r.regs.Rax, nil
|
||||||
|
case x86asm.RCX:
|
||||||
|
return r.regs.Rcx, nil
|
||||||
|
case x86asm.RDX:
|
||||||
|
return r.regs.Rdx, nil
|
||||||
|
case x86asm.RBX:
|
||||||
|
return r.regs.Rbx, nil
|
||||||
|
case x86asm.RSP:
|
||||||
|
return r.regs.Rsp, nil
|
||||||
|
case x86asm.RBP:
|
||||||
|
return r.regs.Rbp, nil
|
||||||
|
case x86asm.RSI:
|
||||||
|
return r.regs.Rsi, nil
|
||||||
|
case x86asm.RDI:
|
||||||
|
return r.regs.Rdi, nil
|
||||||
|
case x86asm.R8:
|
||||||
|
return r.regs.R8, nil
|
||||||
|
case x86asm.R9:
|
||||||
|
return r.regs.R9, nil
|
||||||
|
case x86asm.R10:
|
||||||
|
return r.regs.R10, nil
|
||||||
|
case x86asm.R11:
|
||||||
|
return r.regs.R11, nil
|
||||||
|
case x86asm.R12:
|
||||||
|
return r.regs.R12, nil
|
||||||
|
case x86asm.R13:
|
||||||
|
return r.regs.R13, nil
|
||||||
|
case x86asm.R14:
|
||||||
|
return r.regs.R14, nil
|
||||||
|
case x86asm.R15:
|
||||||
|
return r.regs.R15, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0, UnknownRegisterError
|
||||||
|
}
|
||||||
|
|
||||||
|
func registers(thread *Thread, floatingPoint bool) (Registers, error) {
|
||||||
|
var (
|
||||||
|
regs sys.PtraceRegs
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
thread.dbp.execPtraceFunc(func() { err = sys.PtraceGetRegs(thread.ID, ®s) })
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
r := &Regs{®s, nil}
|
||||||
|
if floatingPoint {
|
||||||
|
r.fpregs, err = thread.fpRegisters()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return r, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// tracks user_fpregs_struct in /usr/include/x86_64-linux-gnu/sys/user.h
|
||||||
|
type PtraceFpRegs struct {
|
||||||
|
Cwd uint16
|
||||||
|
Swd uint16
|
||||||
|
Ftw uint16
|
||||||
|
Fop uint16
|
||||||
|
Rip uint64
|
||||||
|
Rdp uint64
|
||||||
|
Mxcsr uint32
|
||||||
|
MxcrMask uint32
|
||||||
|
StSpace [32]uint32
|
||||||
|
XmmSpace [256]byte
|
||||||
|
padding [24]uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
type PtraceXsave struct {
|
||||||
|
PtraceFpRegs
|
||||||
|
AvxState bool // contains AVX state
|
||||||
|
YmmSpace [256]byte
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
_X86_XSTATE_MAX_SIZE = 2688
|
||||||
|
_NT_X86_XSTATE = 0x202
|
||||||
|
|
||||||
|
_XSAVE_HEADER_START = 512
|
||||||
|
_XSAVE_HEADER_LEN = 64
|
||||||
|
_XSAVE_EXTENDED_REGION_START = 576
|
||||||
|
_XSAVE_SSE_REGION_LEN = 416
|
||||||
|
)
|
||||||
|
|
||||||
|
func (thread *Thread) fpRegisters() (regs []Register, err error) {
|
||||||
|
var fpregs PtraceXsave
|
||||||
|
thread.dbp.execPtraceFunc(func() { fpregs, err = PtraceGetRegset(thread.ID) })
|
||||||
|
|
||||||
|
// x87 registers
|
||||||
|
regs = appendWordReg(regs, "CW", fpregs.Cwd)
|
||||||
|
regs = appendWordReg(regs, "SW", fpregs.Swd)
|
||||||
|
regs = appendWordReg(regs, "TW", fpregs.Ftw)
|
||||||
|
regs = appendWordReg(regs, "FOP", fpregs.Fop)
|
||||||
|
regs = appendQwordReg(regs, "FIP", fpregs.Rip)
|
||||||
|
regs = appendQwordReg(regs, "FDP", fpregs.Rdp)
|
||||||
|
|
||||||
|
for i := 0; i < len(fpregs.StSpace); i += 4 {
|
||||||
|
regs = appendX87Reg(regs, i/4, uint16(fpregs.StSpace[i+2]), uint64(fpregs.StSpace[i+1])<<32|uint64(fpregs.StSpace[i]))
|
||||||
|
}
|
||||||
|
|
||||||
|
// SSE registers
|
||||||
|
regs = appendFlagReg(regs, "MXCSR", uint64(fpregs.Mxcsr), mxcsrDescription, 32)
|
||||||
|
regs = appendDwordReg(regs, "MXCSR_MASK", fpregs.MxcrMask)
|
||||||
|
|
||||||
|
for i := 0; i < len(fpregs.XmmSpace); i += 16 {
|
||||||
|
regs = appendSSEReg(regs, fmt.Sprintf("XMM%d", i/16), fpregs.XmmSpace[i:i+16])
|
||||||
|
if fpregs.AvxState {
|
||||||
|
regs = appendSSEReg(regs, fmt.Sprintf("YMM%d", i/16), fpregs.YmmSpace[i:i+16])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
351
vendor/github.com/derekparker/delve/pkg/proc/registers_windows_amd64.go
generated
vendored
Normal file
351
vendor/github.com/derekparker/delve/pkg/proc/registers_windows_amd64.go
generated
vendored
Normal file
@ -0,0 +1,351 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"rsc.io/x86/x86asm"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Regs represents CPU registers on an AMD64 processor.
|
||||||
|
type Regs struct {
|
||||||
|
rax uint64
|
||||||
|
rbx uint64
|
||||||
|
rcx uint64
|
||||||
|
rdx uint64
|
||||||
|
rdi uint64
|
||||||
|
rsi uint64
|
||||||
|
rbp uint64
|
||||||
|
rsp uint64
|
||||||
|
r8 uint64
|
||||||
|
r9 uint64
|
||||||
|
r10 uint64
|
||||||
|
r11 uint64
|
||||||
|
r12 uint64
|
||||||
|
r13 uint64
|
||||||
|
r14 uint64
|
||||||
|
r15 uint64
|
||||||
|
rip uint64
|
||||||
|
eflags uint64
|
||||||
|
cs uint64
|
||||||
|
fs uint64
|
||||||
|
gs uint64
|
||||||
|
tls uint64
|
||||||
|
fltSave *_XMM_SAVE_AREA32
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Regs) Slice() []Register {
|
||||||
|
var regs = []struct {
|
||||||
|
k string
|
||||||
|
v uint64
|
||||||
|
}{
|
||||||
|
{"Rip", r.rip},
|
||||||
|
{"Rsp", r.rsp},
|
||||||
|
{"Rax", r.rax},
|
||||||
|
{"Rbx", r.rbx},
|
||||||
|
{"Rcx", r.rcx},
|
||||||
|
{"Rdx", r.rdx},
|
||||||
|
{"Rdi", r.rdi},
|
||||||
|
{"Rsi", r.rsi},
|
||||||
|
{"Rbp", r.rbp},
|
||||||
|
{"R8", r.r8},
|
||||||
|
{"R9", r.r9},
|
||||||
|
{"R10", r.r10},
|
||||||
|
{"R11", r.r11},
|
||||||
|
{"R12", r.r12},
|
||||||
|
{"R13", r.r13},
|
||||||
|
{"R14", r.r14},
|
||||||
|
{"R15", r.r15},
|
||||||
|
{"Eflags", r.eflags},
|
||||||
|
{"Cs", r.cs},
|
||||||
|
{"Fs", r.fs},
|
||||||
|
{"Gs", r.gs},
|
||||||
|
{"TLS", r.tls},
|
||||||
|
}
|
||||||
|
outlen := len(regs)
|
||||||
|
if r.fltSave != nil {
|
||||||
|
outlen += 6 + 8 + 2 + 16
|
||||||
|
}
|
||||||
|
out := make([]Register, 0, outlen)
|
||||||
|
for _, reg := range regs {
|
||||||
|
if reg.k == "Eflags" {
|
||||||
|
out = append(out, Register{reg.k, eflagsDescription.Describe(reg.v, 64)})
|
||||||
|
} else {
|
||||||
|
out = appendQwordReg(out, reg.k, reg.v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if r.fltSave != nil {
|
||||||
|
out = appendWordReg(out, "CW", r.fltSave.ControlWord)
|
||||||
|
out = appendWordReg(out, "SW", r.fltSave.StatusWord)
|
||||||
|
out = appendWordReg(out, "TW", uint16(r.fltSave.TagWord))
|
||||||
|
out = appendWordReg(out, "FOP", r.fltSave.ErrorOpcode)
|
||||||
|
out = appendQwordReg(out, "FIP", uint64(r.fltSave.ErrorSelector)<<32|uint64(r.fltSave.ErrorOffset))
|
||||||
|
out = appendQwordReg(out, "FDP", uint64(r.fltSave.DataSelector)<<32|uint64(r.fltSave.DataOffset))
|
||||||
|
|
||||||
|
for i := range r.fltSave.FloatRegisters {
|
||||||
|
out = appendX87Reg(out, i, uint16(r.fltSave.FloatRegisters[i].High), r.fltSave.FloatRegisters[i].Low)
|
||||||
|
}
|
||||||
|
|
||||||
|
out = appendFlagReg(out, "MXCSR", uint64(r.fltSave.MxCsr), mxcsrDescription, 32)
|
||||||
|
out = appendDwordReg(out, "MXCSR_MASK", r.fltSave.MxCsr_Mask)
|
||||||
|
|
||||||
|
for i := 0; i < len(r.fltSave.XmmRegisters); i += 16 {
|
||||||
|
out = appendSSEReg(out, fmt.Sprintf("XMM%d", i/16), r.fltSave.XmmRegisters[i:i+16])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
// PC returns the current program counter
|
||||||
|
// i.e. the RIP CPU register.
|
||||||
|
func (r *Regs) PC() uint64 {
|
||||||
|
return r.rip
|
||||||
|
}
|
||||||
|
|
||||||
|
// SP returns the stack pointer location,
|
||||||
|
// i.e. the RSP register.
|
||||||
|
func (r *Regs) SP() uint64 {
|
||||||
|
return r.rsp
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Regs) BP() uint64 {
|
||||||
|
return r.rbp
|
||||||
|
}
|
||||||
|
|
||||||
|
// CX returns the value of the RCX register.
|
||||||
|
func (r *Regs) CX() uint64 {
|
||||||
|
return r.rcx
|
||||||
|
}
|
||||||
|
|
||||||
|
// TLS returns the value of the register
|
||||||
|
// that contains the location of the thread
|
||||||
|
// local storage segment.
|
||||||
|
func (r *Regs) TLS() uint64 {
|
||||||
|
return r.tls
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetPC sets the RIP register to the value specified by `pc`.
|
||||||
|
func (r *Regs) SetPC(thread *Thread, pc uint64) error {
|
||||||
|
context := newCONTEXT()
|
||||||
|
context.ContextFlags = _CONTEXT_ALL
|
||||||
|
|
||||||
|
err := _GetThreadContext(thread.os.hThread, context)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
context.Rip = pc
|
||||||
|
|
||||||
|
return _SetThreadContext(thread.os.hThread, context)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *Regs) Get(n int) (uint64, error) {
|
||||||
|
reg := x86asm.Reg(n)
|
||||||
|
const (
|
||||||
|
mask8 = 0x000f
|
||||||
|
mask16 = 0x00ff
|
||||||
|
mask32 = 0xffff
|
||||||
|
)
|
||||||
|
|
||||||
|
switch reg {
|
||||||
|
// 8-bit
|
||||||
|
case x86asm.AL:
|
||||||
|
return r.rax & mask8, nil
|
||||||
|
case x86asm.CL:
|
||||||
|
return r.rcx & mask8, nil
|
||||||
|
case x86asm.DL:
|
||||||
|
return r.rdx & mask8, nil
|
||||||
|
case x86asm.BL:
|
||||||
|
return r.rbx & mask8, nil
|
||||||
|
case x86asm.AH:
|
||||||
|
return (r.rax >> 8) & mask8, nil
|
||||||
|
case x86asm.CH:
|
||||||
|
return (r.rcx >> 8) & mask8, nil
|
||||||
|
case x86asm.DH:
|
||||||
|
return (r.rdx >> 8) & mask8, nil
|
||||||
|
case x86asm.BH:
|
||||||
|
return (r.rbx >> 8) & mask8, nil
|
||||||
|
case x86asm.SPB:
|
||||||
|
return r.rsp & mask8, nil
|
||||||
|
case x86asm.BPB:
|
||||||
|
return r.rbp & mask8, nil
|
||||||
|
case x86asm.SIB:
|
||||||
|
return r.rsi & mask8, nil
|
||||||
|
case x86asm.DIB:
|
||||||
|
return r.rdi & mask8, nil
|
||||||
|
case x86asm.R8B:
|
||||||
|
return r.r8 & mask8, nil
|
||||||
|
case x86asm.R9B:
|
||||||
|
return r.r9 & mask8, nil
|
||||||
|
case x86asm.R10B:
|
||||||
|
return r.r10 & mask8, nil
|
||||||
|
case x86asm.R11B:
|
||||||
|
return r.r11 & mask8, nil
|
||||||
|
case x86asm.R12B:
|
||||||
|
return r.r12 & mask8, nil
|
||||||
|
case x86asm.R13B:
|
||||||
|
return r.r13 & mask8, nil
|
||||||
|
case x86asm.R14B:
|
||||||
|
return r.r14 & mask8, nil
|
||||||
|
case x86asm.R15B:
|
||||||
|
return r.r15 & mask8, nil
|
||||||
|
|
||||||
|
// 16-bit
|
||||||
|
case x86asm.AX:
|
||||||
|
return r.rax & mask16, nil
|
||||||
|
case x86asm.CX:
|
||||||
|
return r.rcx & mask16, nil
|
||||||
|
case x86asm.DX:
|
||||||
|
return r.rdx & mask16, nil
|
||||||
|
case x86asm.BX:
|
||||||
|
return r.rbx & mask16, nil
|
||||||
|
case x86asm.SP:
|
||||||
|
return r.rsp & mask16, nil
|
||||||
|
case x86asm.BP:
|
||||||
|
return r.rbp & mask16, nil
|
||||||
|
case x86asm.SI:
|
||||||
|
return r.rsi & mask16, nil
|
||||||
|
case x86asm.DI:
|
||||||
|
return r.rdi & mask16, nil
|
||||||
|
case x86asm.R8W:
|
||||||
|
return r.r8 & mask16, nil
|
||||||
|
case x86asm.R9W:
|
||||||
|
return r.r9 & mask16, nil
|
||||||
|
case x86asm.R10W:
|
||||||
|
return r.r10 & mask16, nil
|
||||||
|
case x86asm.R11W:
|
||||||
|
return r.r11 & mask16, nil
|
||||||
|
case x86asm.R12W:
|
||||||
|
return r.r12 & mask16, nil
|
||||||
|
case x86asm.R13W:
|
||||||
|
return r.r13 & mask16, nil
|
||||||
|
case x86asm.R14W:
|
||||||
|
return r.r14 & mask16, nil
|
||||||
|
case x86asm.R15W:
|
||||||
|
return r.r15 & mask16, nil
|
||||||
|
|
||||||
|
// 32-bit
|
||||||
|
case x86asm.EAX:
|
||||||
|
return r.rax & mask32, nil
|
||||||
|
case x86asm.ECX:
|
||||||
|
return r.rcx & mask32, nil
|
||||||
|
case x86asm.EDX:
|
||||||
|
return r.rdx & mask32, nil
|
||||||
|
case x86asm.EBX:
|
||||||
|
return r.rbx & mask32, nil
|
||||||
|
case x86asm.ESP:
|
||||||
|
return r.rsp & mask32, nil
|
||||||
|
case x86asm.EBP:
|
||||||
|
return r.rbp & mask32, nil
|
||||||
|
case x86asm.ESI:
|
||||||
|
return r.rsi & mask32, nil
|
||||||
|
case x86asm.EDI:
|
||||||
|
return r.rdi & mask32, nil
|
||||||
|
case x86asm.R8L:
|
||||||
|
return r.r8 & mask32, nil
|
||||||
|
case x86asm.R9L:
|
||||||
|
return r.r9 & mask32, nil
|
||||||
|
case x86asm.R10L:
|
||||||
|
return r.r10 & mask32, nil
|
||||||
|
case x86asm.R11L:
|
||||||
|
return r.r11 & mask32, nil
|
||||||
|
case x86asm.R12L:
|
||||||
|
return r.r12 & mask32, nil
|
||||||
|
case x86asm.R13L:
|
||||||
|
return r.r13 & mask32, nil
|
||||||
|
case x86asm.R14L:
|
||||||
|
return r.r14 & mask32, nil
|
||||||
|
case x86asm.R15L:
|
||||||
|
return r.r15 & mask32, nil
|
||||||
|
|
||||||
|
// 64-bit
|
||||||
|
case x86asm.RAX:
|
||||||
|
return r.rax, nil
|
||||||
|
case x86asm.RCX:
|
||||||
|
return r.rcx, nil
|
||||||
|
case x86asm.RDX:
|
||||||
|
return r.rdx, nil
|
||||||
|
case x86asm.RBX:
|
||||||
|
return r.rbx, nil
|
||||||
|
case x86asm.RSP:
|
||||||
|
return r.rsp, nil
|
||||||
|
case x86asm.RBP:
|
||||||
|
return r.rbp, nil
|
||||||
|
case x86asm.RSI:
|
||||||
|
return r.rsi, nil
|
||||||
|
case x86asm.RDI:
|
||||||
|
return r.rdi, nil
|
||||||
|
case x86asm.R8:
|
||||||
|
return r.r8, nil
|
||||||
|
case x86asm.R9:
|
||||||
|
return r.r9, nil
|
||||||
|
case x86asm.R10:
|
||||||
|
return r.r10, nil
|
||||||
|
case x86asm.R11:
|
||||||
|
return r.r11, nil
|
||||||
|
case x86asm.R12:
|
||||||
|
return r.r12, nil
|
||||||
|
case x86asm.R13:
|
||||||
|
return r.r13, nil
|
||||||
|
case x86asm.R14:
|
||||||
|
return r.r14, nil
|
||||||
|
case x86asm.R15:
|
||||||
|
return r.r15, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0, UnknownRegisterError
|
||||||
|
}
|
||||||
|
|
||||||
|
func registers(thread *Thread, floatingPoint bool) (Registers, error) {
|
||||||
|
context := newCONTEXT()
|
||||||
|
|
||||||
|
context.ContextFlags = _CONTEXT_ALL
|
||||||
|
err := _GetThreadContext(thread.os.hThread, context)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var threadInfo _THREAD_BASIC_INFORMATION
|
||||||
|
status := _NtQueryInformationThread(thread.os.hThread, _ThreadBasicInformation, uintptr(unsafe.Pointer(&threadInfo)), uint32(unsafe.Sizeof(threadInfo)), nil)
|
||||||
|
if !_NT_SUCCESS(status) {
|
||||||
|
return nil, fmt.Errorf("NtQueryInformationThread failed: it returns 0x%x", status)
|
||||||
|
}
|
||||||
|
|
||||||
|
regs := &Regs{
|
||||||
|
rax: uint64(context.Rax),
|
||||||
|
rbx: uint64(context.Rbx),
|
||||||
|
rcx: uint64(context.Rcx),
|
||||||
|
rdx: uint64(context.Rdx),
|
||||||
|
rdi: uint64(context.Rdi),
|
||||||
|
rsi: uint64(context.Rsi),
|
||||||
|
rbp: uint64(context.Rbp),
|
||||||
|
rsp: uint64(context.Rsp),
|
||||||
|
r8: uint64(context.R8),
|
||||||
|
r9: uint64(context.R9),
|
||||||
|
r10: uint64(context.R10),
|
||||||
|
r11: uint64(context.R11),
|
||||||
|
r12: uint64(context.R12),
|
||||||
|
r13: uint64(context.R13),
|
||||||
|
r14: uint64(context.R14),
|
||||||
|
r15: uint64(context.R15),
|
||||||
|
rip: uint64(context.Rip),
|
||||||
|
eflags: uint64(context.EFlags),
|
||||||
|
cs: uint64(context.SegCs),
|
||||||
|
fs: uint64(context.SegFs),
|
||||||
|
gs: uint64(context.SegGs),
|
||||||
|
tls: uint64(threadInfo.TebBaseAddress),
|
||||||
|
}
|
||||||
|
|
||||||
|
if floatingPoint {
|
||||||
|
regs.fltSave = &context.FltSave
|
||||||
|
}
|
||||||
|
|
||||||
|
return regs, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (thread *Thread) saveRegisters() (Registers, error) {
|
||||||
|
return nil, fmt.Errorf("not implemented: saveRegisters")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (thread *Thread) restoreRegisters() error {
|
||||||
|
return fmt.Errorf("not implemented: restoreRegisters")
|
||||||
|
}
|
261
vendor/github.com/derekparker/delve/pkg/proc/stack.go
generated
vendored
Normal file
261
vendor/github.com/derekparker/delve/pkg/proc/stack.go
generated
vendored
Normal file
@ -0,0 +1,261 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/derekparker/delve/pkg/dwarf/frame"
|
||||||
|
)
|
||||||
|
|
||||||
|
// This code is partly adaped from runtime.gentraceback in
|
||||||
|
// $GOROOT/src/runtime/traceback.go
|
||||||
|
|
||||||
|
const runtimeStackBarrier = "runtime.stackBarrier"
|
||||||
|
|
||||||
|
// NoReturnAddr is returned when return address
|
||||||
|
// could not be found during stack trace.
|
||||||
|
type NoReturnAddr struct {
|
||||||
|
fn string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (nra NoReturnAddr) Error() string {
|
||||||
|
return fmt.Sprintf("could not find return address for %s", nra.fn)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Stackframe represents a frame in a system stack.
|
||||||
|
type Stackframe struct {
|
||||||
|
// Address the function above this one on the call stack will return to.
|
||||||
|
Current Location
|
||||||
|
// Address of the call instruction for the function above on the call stack.
|
||||||
|
Call Location
|
||||||
|
// Start address of the stack frame.
|
||||||
|
CFA int64
|
||||||
|
// Description of the stack frame.
|
||||||
|
FDE *frame.FrameDescriptionEntry
|
||||||
|
// Return address for this stack frame (as read from the stack frame itself).
|
||||||
|
Ret uint64
|
||||||
|
// Address to the memory location containing the return address
|
||||||
|
addrret uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
// Scope returns a new EvalScope using this frame.
|
||||||
|
func (frame *Stackframe) Scope(thread *Thread) *EvalScope {
|
||||||
|
return &EvalScope{Thread: thread, PC: frame.Current.PC, CFA: frame.CFA}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ReturnAddress returns the return address of the function
|
||||||
|
// this thread is executing.
|
||||||
|
func (t *Thread) ReturnAddress() (uint64, error) {
|
||||||
|
locations, err := t.Stacktrace(2)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
if len(locations) < 2 {
|
||||||
|
return 0, NoReturnAddr{locations[0].Current.Fn.BaseName()}
|
||||||
|
}
|
||||||
|
return locations[1].Current.PC, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *Thread) stackIterator(stkbar []savedLR, stkbarPos int) (*stackIterator, error) {
|
||||||
|
regs, err := t.Registers(false)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return newStackIterator(t.dbp, regs.PC(), regs.SP(), regs.BP(), stkbar, stkbarPos), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Stacktrace returns the stack trace for thread.
|
||||||
|
// Note the locations in the array are return addresses not call addresses.
|
||||||
|
func (t *Thread) Stacktrace(depth int) ([]Stackframe, error) {
|
||||||
|
it, err := t.stackIterator(nil, -1)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return it.stacktrace(depth)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *G) stackIterator() (*stackIterator, error) {
|
||||||
|
stkbar, err := g.stkbar()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if g.thread != nil {
|
||||||
|
return g.thread.stackIterator(stkbar, g.stkbarPos)
|
||||||
|
}
|
||||||
|
return newStackIterator(g.dbp, g.PC, g.SP, 0, stkbar, g.stkbarPos), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Stacktrace returns the stack trace for a goroutine.
|
||||||
|
// Note the locations in the array are return addresses not call addresses.
|
||||||
|
func (g *G) Stacktrace(depth int) ([]Stackframe, error) {
|
||||||
|
it, err := g.stackIterator()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return it.stacktrace(depth)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GoroutineLocation returns the location of the given
|
||||||
|
// goroutine.
|
||||||
|
func (dbp *Process) GoroutineLocation(g *G) *Location {
|
||||||
|
f, l, fn := dbp.PCToLine(g.PC)
|
||||||
|
return &Location{PC: g.PC, File: f, Line: l, Fn: fn}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NullAddrError is an error for a null address.
|
||||||
|
type NullAddrError struct{}
|
||||||
|
|
||||||
|
func (n NullAddrError) Error() string {
|
||||||
|
return "NULL address"
|
||||||
|
}
|
||||||
|
|
||||||
|
// stackIterator holds information
|
||||||
|
// required to iterate and walk the program
|
||||||
|
// stack.
|
||||||
|
type stackIterator struct {
|
||||||
|
pc, sp, bp uint64
|
||||||
|
top bool
|
||||||
|
atend bool
|
||||||
|
frame Stackframe
|
||||||
|
dbp *Process
|
||||||
|
err error
|
||||||
|
|
||||||
|
stackBarrierPC uint64
|
||||||
|
stkbar []savedLR
|
||||||
|
}
|
||||||
|
|
||||||
|
type savedLR struct {
|
||||||
|
ptr uint64
|
||||||
|
val uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
func newStackIterator(dbp *Process, pc, sp, bp uint64, stkbar []savedLR, stkbarPos int) *stackIterator {
|
||||||
|
stackBarrierFunc := dbp.goSymTable.LookupFunc(runtimeStackBarrier) // stack barriers were removed in Go 1.9
|
||||||
|
var stackBarrierPC uint64
|
||||||
|
if stackBarrierFunc != nil && stkbar != nil {
|
||||||
|
stackBarrierPC = stackBarrierFunc.Entry
|
||||||
|
fn := dbp.goSymTable.PCToFunc(pc)
|
||||||
|
if fn != nil && fn.Name == runtimeStackBarrier {
|
||||||
|
// We caught the goroutine as it's executing the stack barrier, we must
|
||||||
|
// determine whether or not g.stackPos has already been incremented or not.
|
||||||
|
if len(stkbar) > 0 && stkbar[stkbarPos].ptr < sp {
|
||||||
|
// runtime.stackBarrier has not incremented stkbarPos.
|
||||||
|
} else if stkbarPos > 0 && stkbar[stkbarPos-1].ptr < sp {
|
||||||
|
// runtime.stackBarrier has incremented stkbarPos.
|
||||||
|
stkbarPos--
|
||||||
|
} else {
|
||||||
|
return &stackIterator{err: fmt.Errorf("failed to unwind through stackBarrier at SP %x", sp)}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
stkbar = stkbar[stkbarPos:]
|
||||||
|
}
|
||||||
|
return &stackIterator{pc: pc, sp: sp, bp: bp, top: true, dbp: dbp, err: nil, atend: false, stackBarrierPC: stackBarrierPC, stkbar: stkbar}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Next points the iterator to the next stack frame.
|
||||||
|
func (it *stackIterator) Next() bool {
|
||||||
|
if it.err != nil || it.atend {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
it.frame, it.err = it.dbp.frameInfo(it.pc, it.sp, it.bp, it.top)
|
||||||
|
if it.err != nil {
|
||||||
|
if _, nofde := it.err.(*frame.NoFDEForPCError); nofde && !it.top {
|
||||||
|
it.frame = Stackframe{Current: Location{PC: it.pc, File: "?", Line: -1}, Call: Location{PC: it.pc, File: "?", Line: -1}, CFA: 0, Ret: 0}
|
||||||
|
it.atend = true
|
||||||
|
it.err = nil
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if it.frame.Ret <= 0 {
|
||||||
|
it.atend = true
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
if it.stkbar != nil && it.frame.Ret == it.stackBarrierPC && it.frame.addrret == it.stkbar[0].ptr {
|
||||||
|
// Skip stack barrier frames
|
||||||
|
it.frame.Ret = it.stkbar[0].val
|
||||||
|
it.stkbar = it.stkbar[1:]
|
||||||
|
}
|
||||||
|
|
||||||
|
// Look for "top of stack" functions.
|
||||||
|
if it.frame.Current.Fn != nil && (it.frame.Current.Fn.Name == "runtime.goexit" || it.frame.Current.Fn.Name == "runtime.rt0_go" || it.frame.Current.Fn.Name == "runtime.mcall") {
|
||||||
|
it.atend = true
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
it.top = false
|
||||||
|
it.pc = it.frame.Ret
|
||||||
|
it.sp = uint64(it.frame.CFA)
|
||||||
|
it.bp, _ = readUintRaw(it.dbp.currentThread, uintptr(it.bp), int64(it.dbp.arch.PtrSize()))
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// Frame returns the frame the iterator is pointing at.
|
||||||
|
func (it *stackIterator) Frame() Stackframe {
|
||||||
|
if it.err != nil {
|
||||||
|
panic(it.err)
|
||||||
|
}
|
||||||
|
return it.frame
|
||||||
|
}
|
||||||
|
|
||||||
|
// Err returns the error encountered during stack iteration.
|
||||||
|
func (it *stackIterator) Err() error {
|
||||||
|
return it.err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) frameInfo(pc, sp, bp uint64, top bool) (Stackframe, error) {
|
||||||
|
fde, err := dbp.frameEntries.FDEForPC(pc)
|
||||||
|
if _, nofde := err.(*frame.NoFDEForPCError); nofde {
|
||||||
|
if bp == 0 {
|
||||||
|
return Stackframe{}, err
|
||||||
|
}
|
||||||
|
// When no FDE is available attempt to use BP instead
|
||||||
|
retaddr := uintptr(int(bp) + dbp.arch.PtrSize())
|
||||||
|
cfa := int64(retaddr) + int64(dbp.arch.PtrSize())
|
||||||
|
return dbp.newStackframe(pc, cfa, retaddr, nil, top)
|
||||||
|
}
|
||||||
|
|
||||||
|
spoffset, retoffset := fde.ReturnAddressOffset(pc)
|
||||||
|
cfa := int64(sp) + spoffset
|
||||||
|
|
||||||
|
retaddr := uintptr(cfa + retoffset)
|
||||||
|
return dbp.newStackframe(pc, cfa, retaddr, fde, top)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) newStackframe(pc uint64, cfa int64, retaddr uintptr, fde *frame.FrameDescriptionEntry, top bool) (Stackframe, error) {
|
||||||
|
if retaddr == 0 {
|
||||||
|
return Stackframe{}, NullAddrError{}
|
||||||
|
}
|
||||||
|
f, l, fn := dbp.PCToLine(pc)
|
||||||
|
ret, err := readUintRaw(dbp.currentThread, retaddr, int64(dbp.arch.PtrSize()))
|
||||||
|
if err != nil {
|
||||||
|
return Stackframe{}, err
|
||||||
|
}
|
||||||
|
r := Stackframe{Current: Location{PC: pc, File: f, Line: l, Fn: fn}, CFA: cfa, FDE: fde, Ret: ret, addrret: uint64(retaddr)}
|
||||||
|
if !top {
|
||||||
|
r.Call.File, r.Call.Line, r.Call.Fn = dbp.PCToLine(pc - 1)
|
||||||
|
r.Call.PC = r.Current.PC
|
||||||
|
} else {
|
||||||
|
r.Call = r.Current
|
||||||
|
}
|
||||||
|
return r, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (it *stackIterator) stacktrace(depth int) ([]Stackframe, error) {
|
||||||
|
if depth < 0 {
|
||||||
|
return nil, errors.New("negative maximum stack depth")
|
||||||
|
}
|
||||||
|
frames := make([]Stackframe, 0, depth+1)
|
||||||
|
for it.Next() {
|
||||||
|
frames = append(frames, it.Frame())
|
||||||
|
if len(frames) >= depth+1 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if err := it.Err(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return frames, nil
|
||||||
|
}
|
112
vendor/github.com/derekparker/delve/pkg/proc/syscall_windows.go
generated
vendored
Normal file
112
vendor/github.com/derekparker/delve/pkg/proc/syscall_windows.go
generated
vendored
Normal file
@ -0,0 +1,112 @@
|
|||||||
|
//go:generate go run $GOROOT/src/syscall/mksyscall_windows.go -output zsyscall_windows.go syscall_windows.go
|
||||||
|
|
||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"syscall"
|
||||||
|
)
|
||||||
|
|
||||||
|
type _NTSTATUS int32
|
||||||
|
|
||||||
|
type _CLIENT_ID struct {
|
||||||
|
UniqueProcess syscall.Handle
|
||||||
|
UniqueThread syscall.Handle
|
||||||
|
}
|
||||||
|
|
||||||
|
type _THREAD_BASIC_INFORMATION struct {
|
||||||
|
ExitStatus _NTSTATUS
|
||||||
|
TebBaseAddress uintptr
|
||||||
|
ClientId _CLIENT_ID
|
||||||
|
AffinityMask uintptr
|
||||||
|
Priority int32
|
||||||
|
BasePriority int32
|
||||||
|
}
|
||||||
|
|
||||||
|
type _CREATE_PROCESS_DEBUG_INFO struct {
|
||||||
|
File syscall.Handle
|
||||||
|
Process syscall.Handle
|
||||||
|
Thread syscall.Handle
|
||||||
|
BaseOfImage uintptr
|
||||||
|
DebugInfoFileOffset uint32
|
||||||
|
DebugInfoSize uint32
|
||||||
|
ThreadLocalBase uintptr
|
||||||
|
StartAddress uintptr
|
||||||
|
ImageName uintptr
|
||||||
|
Unicode uint16
|
||||||
|
}
|
||||||
|
|
||||||
|
type _CREATE_THREAD_DEBUG_INFO struct {
|
||||||
|
Thread syscall.Handle
|
||||||
|
ThreadLocalBase uintptr
|
||||||
|
StartAddress uintptr
|
||||||
|
}
|
||||||
|
|
||||||
|
type _EXIT_PROCESS_DEBUG_INFO struct {
|
||||||
|
ExitCode uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
type _LOAD_DLL_DEBUG_INFO struct {
|
||||||
|
File syscall.Handle
|
||||||
|
BaseOfDll uintptr
|
||||||
|
DebugInfoFileOffset uint32
|
||||||
|
DebugInfoSize uint32
|
||||||
|
ImageName uintptr
|
||||||
|
Unicode uint16
|
||||||
|
}
|
||||||
|
|
||||||
|
type _EXCEPTION_DEBUG_INFO struct {
|
||||||
|
ExceptionRecord _EXCEPTION_RECORD
|
||||||
|
FirstChance uint32
|
||||||
|
}
|
||||||
|
|
||||||
|
type _EXCEPTION_RECORD struct {
|
||||||
|
ExceptionCode uint32
|
||||||
|
ExceptionFlags uint32
|
||||||
|
ExceptionRecord *_EXCEPTION_RECORD
|
||||||
|
ExceptionAddress uintptr
|
||||||
|
NumberParameters uint32
|
||||||
|
ExceptionInformation [_EXCEPTION_MAXIMUM_PARAMETERS]uintptr
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
_ThreadBasicInformation = 0
|
||||||
|
|
||||||
|
_DBG_CONTINUE = 0x00010002
|
||||||
|
_DBG_EXCEPTION_NOT_HANDLED = 0x80010001
|
||||||
|
|
||||||
|
_EXCEPTION_DEBUG_EVENT = 1
|
||||||
|
_CREATE_THREAD_DEBUG_EVENT = 2
|
||||||
|
_CREATE_PROCESS_DEBUG_EVENT = 3
|
||||||
|
_EXIT_THREAD_DEBUG_EVENT = 4
|
||||||
|
_EXIT_PROCESS_DEBUG_EVENT = 5
|
||||||
|
_LOAD_DLL_DEBUG_EVENT = 6
|
||||||
|
_UNLOAD_DLL_DEBUG_EVENT = 7
|
||||||
|
_OUTPUT_DEBUG_STRING_EVENT = 8
|
||||||
|
_RIP_EVENT = 9
|
||||||
|
|
||||||
|
// DEBUG_ONLY_THIS_PROCESS tracks https://msdn.microsoft.com/en-us/library/windows/desktop/ms684863(v=vs.85).aspx
|
||||||
|
_DEBUG_ONLY_THIS_PROCESS = 0x00000002
|
||||||
|
|
||||||
|
_EXCEPTION_BREAKPOINT = 0x80000003
|
||||||
|
_EXCEPTION_SINGLE_STEP = 0x80000004
|
||||||
|
|
||||||
|
_EXCEPTION_MAXIMUM_PARAMETERS = 15
|
||||||
|
)
|
||||||
|
|
||||||
|
func _NT_SUCCESS(x _NTSTATUS) bool {
|
||||||
|
return x >= 0
|
||||||
|
}
|
||||||
|
|
||||||
|
//sys _NtQueryInformationThread(threadHandle syscall.Handle, infoclass int32, info uintptr, infolen uint32, retlen *uint32) (status _NTSTATUS) = ntdll.NtQueryInformationThread
|
||||||
|
//sys _GetThreadContext(thread syscall.Handle, context *_CONTEXT) (err error) = kernel32.GetThreadContext
|
||||||
|
//sys _SetThreadContext(thread syscall.Handle, context *_CONTEXT) (err error) = kernel32.SetThreadContext
|
||||||
|
//sys _SuspendThread(threadid syscall.Handle) (prevsuspcount uint32, err error) [failretval==0xffffffff] = kernel32.SuspendThread
|
||||||
|
//sys _ResumeThread(threadid syscall.Handle) (prevsuspcount uint32, err error) [failretval==0xffffffff] = kernel32.ResumeThread
|
||||||
|
//sys _ContinueDebugEvent(processid uint32, threadid uint32, continuestatus uint32) (err error) = kernel32.ContinueDebugEvent
|
||||||
|
//sys _WriteProcessMemory(process syscall.Handle, baseaddr uintptr, buffer *byte, size uintptr, byteswritten *uintptr) (err error) = kernel32.WriteProcessMemory
|
||||||
|
//sys _ReadProcessMemory(process syscall.Handle, baseaddr uintptr, buffer *byte, size uintptr, bytesread *uintptr) (err error) = kernel32.ReadProcessMemory
|
||||||
|
//sys _DebugBreakProcess(process syscall.Handle) (err error) = kernel32.DebugBreakProcess
|
||||||
|
//sys _WaitForDebugEvent(debugevent *_DEBUG_EVENT, milliseconds uint32) (err error) = kernel32.WaitForDebugEvent
|
||||||
|
//sys _DebugActiveProcess(processid uint32) (err error) = kernel32.DebugActiveProcess
|
||||||
|
//sys _DebugActiveProcessStop(processid uint32) (err error) = kernel32.DebugActiveProcessStop
|
||||||
|
//sys _QueryFullProcessImageName(process syscall.Handle, flags uint32, exename *uint16, size *uint32) (err error) = kernel32.QueryFullProcessImageNameW
|
114
vendor/github.com/derekparker/delve/pkg/proc/syscall_windows_amd64.go
generated
vendored
Normal file
114
vendor/github.com/derekparker/delve/pkg/proc/syscall_windows_amd64.go
generated
vendored
Normal file
@ -0,0 +1,114 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
|
const (
|
||||||
|
_CONTEXT_AMD64 = 0x100000
|
||||||
|
_CONTEXT_CONTROL = (_CONTEXT_AMD64 | 0x1)
|
||||||
|
_CONTEXT_INTEGER = (_CONTEXT_AMD64 | 0x2)
|
||||||
|
_CONTEXT_SEGMENTS = (_CONTEXT_AMD64 | 0x4)
|
||||||
|
_CONTEXT_FLOATING_POINT = (_CONTEXT_AMD64 | 0x8)
|
||||||
|
_CONTEXT_DEBUG_REGISTERS = (_CONTEXT_AMD64 | 0x10)
|
||||||
|
_CONTEXT_FULL = (_CONTEXT_CONTROL | _CONTEXT_INTEGER | _CONTEXT_FLOATING_POINT)
|
||||||
|
_CONTEXT_ALL = (_CONTEXT_CONTROL | _CONTEXT_INTEGER | _CONTEXT_SEGMENTS | _CONTEXT_FLOATING_POINT | _CONTEXT_DEBUG_REGISTERS)
|
||||||
|
_CONTEXT_EXCEPTION_ACTIVE = 0x8000000
|
||||||
|
_CONTEXT_SERVICE_ACTIVE = 0x10000000
|
||||||
|
_CONTEXT_EXCEPTION_REQUEST = 0x40000000
|
||||||
|
_CONTEXT_EXCEPTION_REPORTING = 0x80000000
|
||||||
|
)
|
||||||
|
|
||||||
|
type _M128A struct {
|
||||||
|
Low uint64
|
||||||
|
High int64
|
||||||
|
}
|
||||||
|
|
||||||
|
type _XMM_SAVE_AREA32 struct {
|
||||||
|
ControlWord uint16
|
||||||
|
StatusWord uint16
|
||||||
|
TagWord byte
|
||||||
|
Reserved1 byte
|
||||||
|
ErrorOpcode uint16
|
||||||
|
ErrorOffset uint32
|
||||||
|
ErrorSelector uint16
|
||||||
|
Reserved2 uint16
|
||||||
|
DataOffset uint32
|
||||||
|
DataSelector uint16
|
||||||
|
Reserved3 uint16
|
||||||
|
MxCsr uint32
|
||||||
|
MxCsr_Mask uint32
|
||||||
|
FloatRegisters [8]_M128A
|
||||||
|
XmmRegisters [256]byte
|
||||||
|
Reserved4 [96]byte
|
||||||
|
}
|
||||||
|
|
||||||
|
type _CONTEXT struct {
|
||||||
|
P1Home uint64
|
||||||
|
P2Home uint64
|
||||||
|
P3Home uint64
|
||||||
|
P4Home uint64
|
||||||
|
P5Home uint64
|
||||||
|
P6Home uint64
|
||||||
|
|
||||||
|
ContextFlags uint32
|
||||||
|
MxCsr uint32
|
||||||
|
|
||||||
|
SegCs uint16
|
||||||
|
SegDs uint16
|
||||||
|
SegEs uint16
|
||||||
|
SegFs uint16
|
||||||
|
SegGs uint16
|
||||||
|
SegSs uint16
|
||||||
|
EFlags uint32
|
||||||
|
|
||||||
|
Dr0 uint64
|
||||||
|
Dr1 uint64
|
||||||
|
Dr2 uint64
|
||||||
|
Dr3 uint64
|
||||||
|
Dr6 uint64
|
||||||
|
Dr7 uint64
|
||||||
|
|
||||||
|
Rax uint64
|
||||||
|
Rcx uint64
|
||||||
|
Rdx uint64
|
||||||
|
Rbx uint64
|
||||||
|
Rsp uint64
|
||||||
|
Rbp uint64
|
||||||
|
Rsi uint64
|
||||||
|
Rdi uint64
|
||||||
|
R8 uint64
|
||||||
|
R9 uint64
|
||||||
|
R10 uint64
|
||||||
|
R11 uint64
|
||||||
|
R12 uint64
|
||||||
|
R13 uint64
|
||||||
|
R14 uint64
|
||||||
|
R15 uint64
|
||||||
|
|
||||||
|
Rip uint64
|
||||||
|
|
||||||
|
FltSave _XMM_SAVE_AREA32
|
||||||
|
|
||||||
|
VectorRegister [26]_M128A
|
||||||
|
VectorControl uint64
|
||||||
|
|
||||||
|
DebugControl uint64
|
||||||
|
LastBranchToRip uint64
|
||||||
|
LastBranchFromRip uint64
|
||||||
|
LastExceptionToRip uint64
|
||||||
|
LastExceptionFromRip uint64
|
||||||
|
}
|
||||||
|
|
||||||
|
// newCONTEXT allocates Windows CONTEXT structure aligned to 16 bytes.
|
||||||
|
func newCONTEXT() *_CONTEXT {
|
||||||
|
var c *_CONTEXT
|
||||||
|
buf := make([]byte, unsafe.Sizeof(*c)+15)
|
||||||
|
return (*_CONTEXT)(unsafe.Pointer((uintptr(unsafe.Pointer(&buf[15]))) &^ 15))
|
||||||
|
}
|
||||||
|
|
||||||
|
type _DEBUG_EVENT struct {
|
||||||
|
DebugEventCode uint32
|
||||||
|
ProcessId uint32
|
||||||
|
ThreadId uint32
|
||||||
|
_ uint32 // to align Union properly
|
||||||
|
U [160]byte
|
||||||
|
}
|
510
vendor/github.com/derekparker/delve/pkg/proc/threads.go
generated
vendored
Normal file
510
vendor/github.com/derekparker/delve/pkg/proc/threads.go
generated
vendored
Normal file
@ -0,0 +1,510 @@
|
|||||||
|
package proc
|
||||||
|
|
||||||
|
import (
|
||||||
|
"debug/gosym"
|
||||||
|
"encoding/binary"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"go/ast"
|
||||||
|
"path/filepath"
|
||||||
|
"reflect"
|
||||||
|
"runtime"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"golang.org/x/debug/dwarf"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Thread represents a single thread in the traced process
|
||||||
|
// ID represents the thread id or port, Process holds a reference to the
|
||||||
|
// Process struct that contains info on the process as
|
||||||
|
// a whole, and Status represents the last result of a `wait` call
|
||||||
|
// on this thread.
|
||||||
|
type Thread struct {
|
||||||
|
ID int // Thread ID or mach port
|
||||||
|
Status *WaitStatus // Status returned from last wait call
|
||||||
|
CurrentBreakpoint *Breakpoint // Breakpoint thread is currently stopped at
|
||||||
|
BreakpointConditionMet bool // Output of evaluating the breakpoint's condition
|
||||||
|
BreakpointConditionError error // Error evaluating the breakpoint's condition
|
||||||
|
|
||||||
|
dbp *Process
|
||||||
|
singleStepping bool
|
||||||
|
running bool
|
||||||
|
os *OSSpecificDetails
|
||||||
|
}
|
||||||
|
|
||||||
|
// Location represents the location of a thread.
|
||||||
|
// Holds information on the current instruction
|
||||||
|
// address, the source file:line, and the function.
|
||||||
|
type Location struct {
|
||||||
|
PC uint64
|
||||||
|
File string
|
||||||
|
Line int
|
||||||
|
Fn *gosym.Func
|
||||||
|
}
|
||||||
|
|
||||||
|
// Continue the execution of this thread.
|
||||||
|
//
|
||||||
|
// If we are currently at a breakpoint, we'll clear it
|
||||||
|
// first and then resume execution. Thread will continue until
|
||||||
|
// it hits a breakpoint or is signaled.
|
||||||
|
func (thread *Thread) Continue() error {
|
||||||
|
pc, err := thread.PC()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
// Check whether we are stopped at a breakpoint, and
|
||||||
|
// if so, single step over it before continuing.
|
||||||
|
if _, ok := thread.dbp.FindBreakpoint(pc); ok {
|
||||||
|
if err := thread.StepInstruction(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return thread.resume()
|
||||||
|
}
|
||||||
|
|
||||||
|
// StepInstruction steps a single instruction.
|
||||||
|
//
|
||||||
|
// Executes exactly one instruction and then returns.
|
||||||
|
// If the thread is at a breakpoint, we first clear it,
|
||||||
|
// execute the instruction, and then replace the breakpoint.
|
||||||
|
// Otherwise we simply execute the next instruction.
|
||||||
|
func (thread *Thread) StepInstruction() (err error) {
|
||||||
|
thread.running = true
|
||||||
|
thread.singleStepping = true
|
||||||
|
defer func() {
|
||||||
|
thread.singleStepping = false
|
||||||
|
thread.running = false
|
||||||
|
}()
|
||||||
|
pc, err := thread.PC()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
bp, ok := thread.dbp.FindBreakpoint(pc)
|
||||||
|
if ok {
|
||||||
|
// Clear the breakpoint so that we can continue execution.
|
||||||
|
_, err = bp.Clear(thread)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Restore breakpoint now that we have passed it.
|
||||||
|
defer func() {
|
||||||
|
err = thread.dbp.writeSoftwareBreakpoint(thread, bp.Addr)
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
|
||||||
|
err = thread.singleStep()
|
||||||
|
if err != nil {
|
||||||
|
if _, exited := err.(ProcessExitedError); exited {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return fmt.Errorf("step failed: %s", err.Error())
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Location returns the threads location, including the file:line
|
||||||
|
// of the corresponding source code, the function we're in
|
||||||
|
// and the current instruction address.
|
||||||
|
func (thread *Thread) Location() (*Location, error) {
|
||||||
|
pc, err := thread.PC()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
f, l, fn := thread.dbp.PCToLine(pc)
|
||||||
|
return &Location{PC: pc, File: f, Line: l, Fn: fn}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ThreadBlockedError is returned when the thread
|
||||||
|
// is blocked in the scheduler.
|
||||||
|
type ThreadBlockedError struct{}
|
||||||
|
|
||||||
|
func (tbe ThreadBlockedError) Error() string {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// returns topmost frame of g or thread if g is nil
|
||||||
|
func topframe(g *G, thread *Thread) (Stackframe, error) {
|
||||||
|
var frames []Stackframe
|
||||||
|
var err error
|
||||||
|
|
||||||
|
if g == nil {
|
||||||
|
if thread.blocked() {
|
||||||
|
return Stackframe{}, ThreadBlockedError{}
|
||||||
|
}
|
||||||
|
frames, err = thread.Stacktrace(0)
|
||||||
|
} else {
|
||||||
|
frames, err = g.Stacktrace(0)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return Stackframe{}, err
|
||||||
|
}
|
||||||
|
if len(frames) < 1 {
|
||||||
|
return Stackframe{}, errors.New("empty stack trace")
|
||||||
|
}
|
||||||
|
return frames[0], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set breakpoints at every line, and the return address. Also look for
|
||||||
|
// a deferred function and set a breakpoint there too.
|
||||||
|
// If stepInto is true it will also set breakpoints inside all
|
||||||
|
// functions called on the current source line, for non-absolute CALLs
|
||||||
|
// a breakpoint of kind StepBreakpoint is set on the CALL instruction,
|
||||||
|
// Continue will take care of setting a breakpoint to the destination
|
||||||
|
// once the CALL is reached.
|
||||||
|
func (dbp *Process) next(stepInto bool) error {
|
||||||
|
topframe, err := topframe(dbp.selectedGoroutine, dbp.currentThread)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
success := false
|
||||||
|
defer func() {
|
||||||
|
if !success {
|
||||||
|
dbp.ClearInternalBreakpoints()
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
csource := filepath.Ext(topframe.Current.File) != ".go"
|
||||||
|
thread := dbp.currentThread
|
||||||
|
currentGoroutine := false
|
||||||
|
if dbp.selectedGoroutine != nil && dbp.selectedGoroutine.thread != nil {
|
||||||
|
thread = dbp.selectedGoroutine.thread
|
||||||
|
currentGoroutine = true
|
||||||
|
}
|
||||||
|
|
||||||
|
text, err := thread.Disassemble(topframe.FDE.Begin(), topframe.FDE.End(), currentGoroutine)
|
||||||
|
if err != nil && stepInto {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
cond := sameGoroutineCondition(dbp.selectedGoroutine)
|
||||||
|
|
||||||
|
if stepInto {
|
||||||
|
for _, instr := range text {
|
||||||
|
if instr.Loc.File != topframe.Current.File || instr.Loc.Line != topframe.Current.Line || !instr.IsCall() {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if instr.DestLoc != nil && instr.DestLoc.Fn != nil {
|
||||||
|
if err := dbp.setStepIntoBreakpoint([]AsmInstruction{instr}, cond); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Non-absolute call instruction, set a StepBreakpoint here
|
||||||
|
if _, err := dbp.SetBreakpoint(instr.Loc.PC, StepBreakpoint, cond); err != nil {
|
||||||
|
if _, ok := err.(BreakpointExistsError); !ok {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !csource {
|
||||||
|
deferreturns := []uint64{}
|
||||||
|
|
||||||
|
// Find all runtime.deferreturn locations in the function
|
||||||
|
// See documentation of Breakpoint.DeferCond for why this is necessary
|
||||||
|
for _, instr := range text {
|
||||||
|
if instr.IsCall() && instr.DestLoc != nil && instr.DestLoc.Fn != nil && instr.DestLoc.Fn.Name == "runtime.deferreturn" {
|
||||||
|
deferreturns = append(deferreturns, instr.Loc.PC)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set breakpoint on the most recently deferred function (if any)
|
||||||
|
var deferpc uint64 = 0
|
||||||
|
if dbp.selectedGoroutine != nil {
|
||||||
|
deferPCEntry := dbp.selectedGoroutine.DeferPC()
|
||||||
|
if deferPCEntry != 0 {
|
||||||
|
_, _, deferfn := dbp.goSymTable.PCToLine(deferPCEntry)
|
||||||
|
var err error
|
||||||
|
deferpc, err = dbp.FirstPCAfterPrologue(deferfn, false)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if deferpc != 0 && deferpc != topframe.Current.PC {
|
||||||
|
bp, err := dbp.SetBreakpoint(deferpc, NextDeferBreakpoint, cond)
|
||||||
|
if err != nil {
|
||||||
|
if _, ok := err.(BreakpointExistsError); !ok {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if bp != nil {
|
||||||
|
bp.DeferReturns = deferreturns
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add breakpoints on all the lines in the current function
|
||||||
|
pcs, err := dbp.lineInfo.AllPCsBetween(topframe.FDE.Begin(), topframe.FDE.End()-1, topframe.Current.File)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
if !csource {
|
||||||
|
var covered bool
|
||||||
|
for i := range pcs {
|
||||||
|
if topframe.FDE.Cover(pcs[i]) {
|
||||||
|
covered = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !covered {
|
||||||
|
fn := dbp.goSymTable.PCToFunc(topframe.Ret)
|
||||||
|
if dbp.selectedGoroutine != nil && fn != nil && fn.Name == "runtime.goexit" {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add a breakpoint on the return address for the current frame
|
||||||
|
pcs = append(pcs, topframe.Ret)
|
||||||
|
success = true
|
||||||
|
return dbp.setInternalBreakpoints(topframe.Current.PC, pcs, NextBreakpoint, cond)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbp *Process) setStepIntoBreakpoint(text []AsmInstruction, cond ast.Expr) error {
|
||||||
|
if len(text) <= 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
instr := text[0]
|
||||||
|
|
||||||
|
if instr.DestLoc == nil || instr.DestLoc.Fn == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
fn := instr.DestLoc.Fn
|
||||||
|
|
||||||
|
// Ensure PC and Entry match, otherwise StepInto is likely to set
|
||||||
|
// its breakpoint before DestLoc.PC and hence run too far ahead.
|
||||||
|
// Calls to runtime.duffzero and duffcopy have this problem.
|
||||||
|
if fn.Entry != instr.DestLoc.PC {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Skip unexported runtime functions
|
||||||
|
if strings.HasPrefix(fn.Name, "runtime.") && !isExportedRuntime(fn.Name) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
//TODO(aarzilli): if we want to let users hide functions
|
||||||
|
// or entire packages from being stepped into with 'step'
|
||||||
|
// those extra checks should be done here.
|
||||||
|
|
||||||
|
// Set a breakpoint after the function's prologue
|
||||||
|
pc, _ := dbp.FirstPCAfterPrologue(fn, false)
|
||||||
|
if _, err := dbp.SetBreakpoint(pc, NextBreakpoint, cond); err != nil {
|
||||||
|
if _, ok := err.(BreakpointExistsError); !ok {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// setInternalBreakpoints sets a breakpoint to all addresses specified in pcs
|
||||||
|
// skipping over curpc and curpc-1
|
||||||
|
func (dbp *Process) setInternalBreakpoints(curpc uint64, pcs []uint64, kind BreakpointKind, cond ast.Expr) error {
|
||||||
|
for i := range pcs {
|
||||||
|
if pcs[i] == curpc || pcs[i] == curpc-1 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if _, err := dbp.SetBreakpoint(pcs[i], kind, cond); err != nil {
|
||||||
|
if _, ok := err.(BreakpointExistsError); !ok {
|
||||||
|
dbp.ClearInternalBreakpoints()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetPC sets the PC for this thread.
|
||||||
|
func (thread *Thread) SetPC(pc uint64) error {
|
||||||
|
regs, err := thread.Registers(false)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return regs.SetPC(thread, pc)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (thread *Thread) getGVariable() (*Variable, error) {
|
||||||
|
regs, err := thread.Registers(false)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if thread.dbp.arch.GStructOffset() == 0 {
|
||||||
|
// GetG was called through SwitchThread / updateThreadList during initialization
|
||||||
|
// thread.dbp.arch isn't setup yet (it needs a current thread to read global variables from)
|
||||||
|
return nil, fmt.Errorf("g struct offset not initialized")
|
||||||
|
}
|
||||||
|
|
||||||
|
gaddrbs, err := thread.readMemory(uintptr(regs.TLS()+thread.dbp.arch.GStructOffset()), thread.dbp.arch.PtrSize())
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
gaddr := uintptr(binary.LittleEndian.Uint64(gaddrbs))
|
||||||
|
|
||||||
|
// On Windows, the value at TLS()+GStructOffset() is a
|
||||||
|
// pointer to the G struct.
|
||||||
|
needsDeref := runtime.GOOS == "windows"
|
||||||
|
|
||||||
|
return thread.newGVariable(gaddr, needsDeref)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (thread *Thread) newGVariable(gaddr uintptr, deref bool) (*Variable, error) {
|
||||||
|
typ, err := thread.dbp.findType("runtime.g")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
name := ""
|
||||||
|
|
||||||
|
if deref {
|
||||||
|
typ = &dwarf.PtrType{dwarf.CommonType{int64(thread.dbp.arch.PtrSize()), "", reflect.Ptr, 0}, typ}
|
||||||
|
} else {
|
||||||
|
name = "runtime.curg"
|
||||||
|
}
|
||||||
|
|
||||||
|
return thread.newVariable(name, gaddr, typ), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetG returns information on the G (goroutine) that is executing on this thread.
|
||||||
|
//
|
||||||
|
// The G structure for a thread is stored in thread local storage. Here we simply
|
||||||
|
// calculate the address and read and parse the G struct.
|
||||||
|
//
|
||||||
|
// We cannot simply use the allg linked list in order to find the M that represents
|
||||||
|
// the given OS thread and follow its G pointer because on Darwin mach ports are not
|
||||||
|
// universal, so our port for this thread would not map to the `id` attribute of the M
|
||||||
|
// structure. Also, when linked against libc, Go prefers the libc version of clone as
|
||||||
|
// opposed to the runtime version. This has the consequence of not setting M.id for
|
||||||
|
// any thread, regardless of OS.
|
||||||
|
//
|
||||||
|
// In order to get around all this craziness, we read the address of the G structure for
|
||||||
|
// the current thread from the thread local storage area.
|
||||||
|
func (thread *Thread) GetG() (g *G, err error) {
|
||||||
|
gaddr, err := thread.getGVariable()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
g, err = gaddr.parseG()
|
||||||
|
if err == nil {
|
||||||
|
g.thread = thread
|
||||||
|
if loc, err := thread.Location(); err == nil {
|
||||||
|
g.CurrentLoc = *loc
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Stopped returns whether the thread is stopped at
|
||||||
|
// the operating system level. Actual implementation
|
||||||
|
// is OS dependant, look in OS thread file.
|
||||||
|
func (thread *Thread) Stopped() bool {
|
||||||
|
return thread.stopped()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Halt stops this thread from executing. Actual
|
||||||
|
// implementation is OS dependant. Look in OS
|
||||||
|
// thread file.
|
||||||
|
func (thread *Thread) Halt() (err error) {
|
||||||
|
defer func() {
|
||||||
|
if err == nil {
|
||||||
|
thread.running = false
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
if thread.Stopped() {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
err = thread.halt()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Scope returns the current EvalScope for this thread.
|
||||||
|
func (thread *Thread) Scope() (*EvalScope, error) {
|
||||||
|
locations, err := thread.Stacktrace(0)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if len(locations) < 1 {
|
||||||
|
return nil, errors.New("could not decode first frame")
|
||||||
|
}
|
||||||
|
return locations[0].Scope(thread), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetCurrentBreakpoint sets the current breakpoint that this
|
||||||
|
// thread is stopped at as CurrentBreakpoint on the thread struct.
|
||||||
|
func (thread *Thread) SetCurrentBreakpoint() error {
|
||||||
|
thread.CurrentBreakpoint = nil
|
||||||
|
pc, err := thread.PC()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if bp, ok := thread.dbp.FindBreakpoint(pc); ok {
|
||||||
|
thread.CurrentBreakpoint = bp
|
||||||
|
if err = thread.SetPC(bp.Addr); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
thread.BreakpointConditionMet, thread.BreakpointConditionError = bp.checkCondition(thread)
|
||||||
|
if thread.onTriggeredBreakpoint() {
|
||||||
|
if g, err := thread.GetG(); err == nil {
|
||||||
|
thread.CurrentBreakpoint.HitCount[g.ID]++
|
||||||
|
}
|
||||||
|
thread.CurrentBreakpoint.TotalHitCount++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (thread *Thread) clearBreakpointState() {
|
||||||
|
thread.CurrentBreakpoint = nil
|
||||||
|
thread.BreakpointConditionMet = false
|
||||||
|
thread.BreakpointConditionError = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (thread *Thread) onTriggeredBreakpoint() bool {
|
||||||
|
return (thread.CurrentBreakpoint != nil) && thread.BreakpointConditionMet
|
||||||
|
}
|
||||||
|
|
||||||
|
func (thread *Thread) onTriggeredInternalBreakpoint() bool {
|
||||||
|
return thread.onTriggeredBreakpoint() && thread.CurrentBreakpoint.Internal()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (thread *Thread) onRuntimeBreakpoint() bool {
|
||||||
|
loc, err := thread.Location()
|
||||||
|
if err != nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return loc.Fn != nil && loc.Fn.Name == "runtime.breakpoint"
|
||||||
|
}
|
||||||
|
|
||||||
|
// onNextGorutine returns true if this thread is on the goroutine requested by the current 'next' command
|
||||||
|
func (thread *Thread) onNextGoroutine() (bool, error) {
|
||||||
|
var bp *Breakpoint
|
||||||
|
for i := range thread.dbp.breakpoints {
|
||||||
|
if thread.dbp.breakpoints[i].Internal() {
|
||||||
|
bp = thread.dbp.breakpoints[i]
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if bp == nil {
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
if bp.Kind == NextDeferBreakpoint {
|
||||||
|
// we just want to check the condition on the goroutine id here
|
||||||
|
bp.Kind = NextBreakpoint
|
||||||
|
defer func() {
|
||||||
|
bp.Kind = NextDeferBreakpoint
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
return bp.checkCondition(thread)
|
||||||
|
}
|
177
vendor/github.com/derekparker/delve/pkg/proc/threads_darwin.c
generated
vendored
Normal file
177
vendor/github.com/derekparker/delve/pkg/proc/threads_darwin.c
generated
vendored
Normal file
@ -0,0 +1,177 @@
|
|||||||
|
#include "threads_darwin.h"
|
||||||
|
|
||||||
|
int
|
||||||
|
write_memory(task_t task, mach_vm_address_t addr, void *d, mach_msg_type_number_t len) {
|
||||||
|
kern_return_t kret;
|
||||||
|
vm_region_submap_short_info_data_64_t info;
|
||||||
|
mach_msg_type_number_t count = VM_REGION_SUBMAP_SHORT_INFO_COUNT_64;
|
||||||
|
mach_vm_size_t l = len;
|
||||||
|
mach_port_t objname;
|
||||||
|
|
||||||
|
if (len == 1) l = 2;
|
||||||
|
kret = mach_vm_region((vm_map_t)task, &(mach_vm_address_t){addr}, (mach_vm_size_t*)&l, VM_REGION_BASIC_INFO_64, (vm_region_info_t)&info, &count, &objname);
|
||||||
|
if (kret != KERN_SUCCESS) return -1;
|
||||||
|
|
||||||
|
// Set permissions to enable writting to this memory location
|
||||||
|
kret = mach_vm_protect(task, addr, len, FALSE, VM_PROT_WRITE|VM_PROT_COPY|VM_PROT_READ);
|
||||||
|
if (kret != KERN_SUCCESS) return -1;
|
||||||
|
|
||||||
|
kret = mach_vm_write((vm_map_t)task, addr, (vm_offset_t)d, len);
|
||||||
|
if (kret != KERN_SUCCESS) return -1;
|
||||||
|
|
||||||
|
// Restore virtual memory permissions
|
||||||
|
kret = mach_vm_protect(task, addr, len, FALSE, info.protection);
|
||||||
|
if (kret != KERN_SUCCESS) return -1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
read_memory(task_t task, mach_vm_address_t addr, void *d, mach_msg_type_number_t len) {
|
||||||
|
kern_return_t kret;
|
||||||
|
pointer_t data;
|
||||||
|
mach_msg_type_number_t count;
|
||||||
|
|
||||||
|
kret = mach_vm_read((vm_map_t)task, addr, len, &data, &count);
|
||||||
|
if (kret != KERN_SUCCESS) return -1;
|
||||||
|
memcpy(d, (void *)data, len);
|
||||||
|
|
||||||
|
kret = vm_deallocate(task, data, len);
|
||||||
|
if (kret != KERN_SUCCESS) return -1;
|
||||||
|
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
get_registers(mach_port_name_t task, x86_thread_state64_t *state) {
|
||||||
|
kern_return_t kret;
|
||||||
|
mach_msg_type_number_t stateCount = x86_THREAD_STATE64_COUNT;
|
||||||
|
// TODO(dp) - possible memory leak - vm_deallocate state
|
||||||
|
return thread_get_state(task, x86_THREAD_STATE64, (thread_state_t)state, &stateCount);
|
||||||
|
}
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
get_fpu_registers(mach_port_name_t task, x86_float_state64_t *state) {
|
||||||
|
kern_return_t kret;
|
||||||
|
mach_msg_type_number_t stateCount = x86_FLOAT_STATE64_COUNT;
|
||||||
|
return thread_get_state(task, x86_FLOAT_STATE64, (thread_state_t)state, &stateCount);
|
||||||
|
}
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
get_identity(mach_port_name_t task, thread_identifier_info_data_t *idinfo) {
|
||||||
|
mach_msg_type_number_t idinfoCount = THREAD_IDENTIFIER_INFO_COUNT;
|
||||||
|
return thread_info(task, THREAD_IDENTIFIER_INFO, (thread_info_t)idinfo, &idinfoCount);
|
||||||
|
}
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
set_registers(mach_port_name_t task, x86_thread_state64_t *state) {
|
||||||
|
mach_msg_type_number_t stateCount = x86_THREAD_STATE64_COUNT;
|
||||||
|
return thread_set_state(task, x86_THREAD_STATE64, (thread_state_t)state, stateCount);
|
||||||
|
}
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
set_pc(thread_act_t task, uint64_t pc) {
|
||||||
|
kern_return_t kret;
|
||||||
|
x86_thread_state64_t state;
|
||||||
|
mach_msg_type_number_t stateCount = x86_THREAD_STATE64_COUNT;
|
||||||
|
|
||||||
|
kret = thread_get_state(task, x86_THREAD_STATE64, (thread_state_t)&state, &stateCount);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
state.__rip = pc;
|
||||||
|
|
||||||
|
return thread_set_state(task, x86_THREAD_STATE64, (thread_state_t)&state, stateCount);
|
||||||
|
}
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
single_step(thread_act_t thread) {
|
||||||
|
kern_return_t kret;
|
||||||
|
x86_thread_state64_t regs;
|
||||||
|
mach_msg_type_number_t count = x86_THREAD_STATE64_COUNT;
|
||||||
|
|
||||||
|
kret = thread_get_state(thread, x86_THREAD_STATE64, (thread_state_t)®s, &count);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
// Set trap bit in rflags
|
||||||
|
regs.__rflags |= 0x100UL;
|
||||||
|
|
||||||
|
kret = thread_set_state(thread, x86_THREAD_STATE64, (thread_state_t)®s, count);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
return resume_thread(thread);
|
||||||
|
}
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
resume_thread(thread_act_t thread) {
|
||||||
|
kern_return_t kret;
|
||||||
|
struct thread_basic_info info;
|
||||||
|
unsigned int info_count = THREAD_BASIC_INFO_COUNT;
|
||||||
|
|
||||||
|
kret = thread_info((thread_t)thread, THREAD_BASIC_INFO, (thread_info_t)&info, &info_count);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
for (int i = 0; i < info.suspend_count; i++) {
|
||||||
|
kret = thread_resume(thread);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
}
|
||||||
|
return KERN_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
kern_return_t
|
||||||
|
clear_trap_flag(thread_act_t thread) {
|
||||||
|
kern_return_t kret;
|
||||||
|
x86_thread_state64_t regs;
|
||||||
|
mach_msg_type_number_t count = x86_THREAD_STATE64_COUNT;
|
||||||
|
|
||||||
|
kret = thread_get_state(thread, x86_THREAD_STATE64, (thread_state_t)®s, &count);
|
||||||
|
if (kret != KERN_SUCCESS) return kret;
|
||||||
|
|
||||||
|
// Clear trap bit in rflags
|
||||||
|
regs.__rflags ^= 0x100UL;
|
||||||
|
|
||||||
|
return thread_set_state(thread, x86_THREAD_STATE64, (thread_state_t)®s, count);
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
thread_blocked(thread_act_t thread) {
|
||||||
|
kern_return_t kret;
|
||||||
|
struct thread_basic_info info;
|
||||||
|
unsigned int info_count = THREAD_BASIC_INFO_COUNT;
|
||||||
|
|
||||||
|
kret = thread_info((thread_t)thread, THREAD_BASIC_INFO, (thread_info_t)&info, &info_count);
|
||||||
|
if (kret != KERN_SUCCESS) return -1;
|
||||||
|
|
||||||
|
return info.suspend_count;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
num_running_threads(task_t task) {
|
||||||
|
kern_return_t kret;
|
||||||
|
thread_act_array_t list;
|
||||||
|
mach_msg_type_number_t count;
|
||||||
|
int i, n = 0;
|
||||||
|
|
||||||
|
kret = task_threads(task, &list, &count);
|
||||||
|
if (kret != KERN_SUCCESS) {
|
||||||
|
return -kret;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 0; i < count; ++i) {
|
||||||
|
thread_act_t thread = list[i];
|
||||||
|
struct thread_basic_info info;
|
||||||
|
unsigned int info_count = THREAD_BASIC_INFO_COUNT;
|
||||||
|
|
||||||
|
kret = thread_info((thread_t)thread, THREAD_BASIC_INFO, (thread_info_t)&info, &info_count);
|
||||||
|
|
||||||
|
if (kret == KERN_SUCCESS) {
|
||||||
|
if (info.suspend_count == 0) {
|
||||||
|
++n;
|
||||||
|
} else {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
kret = vm_deallocate(mach_task_self(), (vm_address_t) list, count * sizeof(list[0]));
|
||||||
|
if (kret != KERN_SUCCESS) return -kret;
|
||||||
|
|
||||||
|
return n;
|
||||||
|
}
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user