2016-03-04 17:09:08 -08:00
|
|
|
// Copyright 2016 The Go Authors. All rights reserved.
|
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
|
|
package syntax
|
|
|
|
|
|
|
|
|
|
import (
|
2016-10-31 16:58:15 -07:00
|
|
|
"errors"
|
2016-03-04 17:09:08 -08:00
|
|
|
"fmt"
|
|
|
|
|
"io"
|
|
|
|
|
"os"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type Mode uint
|
|
|
|
|
|
cmd/compile: add go:notinheap type pragma
This adds a //go:notinheap pragma for declarations of types that must
not be heap allocated. We ensure these rules by disallowing new(T),
make([]T), append([]T), or implicit allocation of T, by disallowing
conversions to notinheap types, and by propagating notinheap to any
struct or array that contains notinheap elements.
The utility of this pragma is that we can eliminate write barriers for
writes to pointers to go:notinheap types, since the write barrier is
guaranteed to be a no-op. This will let us mark several scheduler and
memory allocator structures as go:notinheap, which will let us
disallow write barriers in the scheduler and memory allocator much
more thoroughly and also eliminate some problematic hybrid write
barriers.
This also makes go:nowritebarrierrec and go:yeswritebarrierrec much
more powerful. Currently we use go:nowritebarrier all over the place,
but it's almost never what you actually want: when write barriers are
illegal, they're typically illegal for a whole dynamic scope. Partly
this is because go:nowritebarrier has been around longer, but it's
also because go:nowritebarrierrec couldn't be used in situations that
had no-op write barriers or where some nested scope did allow write
barriers. go:notinheap eliminates many no-op write barriers and
go:yeswritebarrierrec makes it possible to opt back in to write
barriers, so these two changes will let us use go:nowritebarrierrec
far more liberally.
This updates #13386, which is about controlling pointers from non-GC'd
memory to GC'd memory. That would require some additional pragma (or
pragmas), but could build on this pragma.
Change-Id: I6314f8f4181535dd166887c9ec239977b54940bd
Reviewed-on: https://go-review.googlesource.com/30939
Reviewed-by: Keith Randall <khr@golang.org>
Reviewed-by: Matthew Dempsky <mdempsky@google.com>
2016-10-11 22:53:27 -04:00
|
|
|
// A Pragma value is a set of flags that augment a function or
|
|
|
|
|
// type declaration. Callers may assign meaning to the flags as
|
2016-08-30 16:31:53 -07:00
|
|
|
// appropriate.
|
|
|
|
|
type Pragma uint16
|
|
|
|
|
|
2016-03-04 17:09:08 -08:00
|
|
|
type ErrorHandler func(pos, line int, msg string)
|
|
|
|
|
|
2016-08-30 16:31:53 -07:00
|
|
|
// A PragmaHandler is used to process //line and //go: directives as
|
|
|
|
|
// they're scanned. The returned Pragma value will be unioned into the
|
|
|
|
|
// next FuncDecl node.
|
|
|
|
|
type PragmaHandler func(pos, line int, text string) Pragma
|
|
|
|
|
|
2016-03-04 17:09:08 -08:00
|
|
|
// TODO(gri) These need a lot more work.
|
|
|
|
|
|
2016-08-30 16:31:53 -07:00
|
|
|
func ReadFile(filename string, errh ErrorHandler, pragh PragmaHandler, mode Mode) (*File, error) {
|
2016-03-04 17:09:08 -08:00
|
|
|
src, err := os.Open(filename)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
defer src.Close()
|
2016-08-30 16:31:53 -07:00
|
|
|
return Read(src, errh, pragh, mode)
|
2016-03-04 17:09:08 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type bytesReader struct {
|
|
|
|
|
data []byte
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (r *bytesReader) Read(p []byte) (int, error) {
|
|
|
|
|
if len(r.data) > 0 {
|
|
|
|
|
n := copy(p, r.data)
|
|
|
|
|
r.data = r.data[n:]
|
|
|
|
|
return n, nil
|
|
|
|
|
}
|
|
|
|
|
return 0, io.EOF
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-30 16:31:53 -07:00
|
|
|
func ReadBytes(src []byte, errh ErrorHandler, pragh PragmaHandler, mode Mode) (*File, error) {
|
|
|
|
|
return Read(&bytesReader{src}, errh, pragh, mode)
|
2016-03-04 17:09:08 -08:00
|
|
|
}
|
|
|
|
|
|
2016-10-31 16:58:15 -07:00
|
|
|
func Read(src io.Reader, errh ErrorHandler, pragh PragmaHandler, mode Mode) (ast *File, err error) {
|
|
|
|
|
defer func() {
|
|
|
|
|
if p := recover(); p != nil {
|
|
|
|
|
if msg, ok := p.(parserError); ok {
|
|
|
|
|
err = errors.New(string(msg))
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
panic(p)
|
|
|
|
|
}
|
|
|
|
|
}()
|
|
|
|
|
|
2016-03-04 17:09:08 -08:00
|
|
|
var p parser
|
2016-08-30 16:31:53 -07:00
|
|
|
p.init(src, errh, pragh)
|
2016-03-04 17:09:08 -08:00
|
|
|
p.next()
|
2016-10-31 16:58:15 -07:00
|
|
|
ast = p.file()
|
2016-03-04 17:09:08 -08:00
|
|
|
|
2016-10-31 16:58:15 -07:00
|
|
|
// TODO(gri) This isn't quite right: Even if there's an error handler installed
|
|
|
|
|
// we should report an error if parsing found syntax errors. This also
|
|
|
|
|
// requires updating the noder's ReadFile call.
|
2016-03-04 17:09:08 -08:00
|
|
|
if errh == nil && p.nerrors > 0 {
|
2016-10-31 16:58:15 -07:00
|
|
|
ast = nil
|
|
|
|
|
err = fmt.Errorf("%d syntax errors", p.nerrors)
|
2016-03-04 17:09:08 -08:00
|
|
|
}
|
|
|
|
|
|
2016-10-31 16:58:15 -07:00
|
|
|
return
|
2016-03-04 17:09:08 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func Write(w io.Writer, n *File) error {
|
|
|
|
|
panic("unimplemented")
|
|
|
|
|
}
|