Commit 1583931b authored by Anthony Martin's avatar Anthony Martin

syscall: cache environment variables on Plan 9.

This can drastically reduce the number of system
calls made by programs that repeatedly query the
environment.

R=golang-dev, rsc
CC=golang-dev
https://golang.org/cl/5599054
parent 82555d7b
...@@ -6,69 +6,123 @@ ...@@ -6,69 +6,123 @@
package syscall package syscall
import "errors" import (
"errors"
"sync"
)
func Getenv(key string) (value string, found bool) { var (
if len(key) == 0 { // envOnce guards initialization by copyenv, which populates env.
return "", false envOnce sync.Once
// envLock guards env.
envLock sync.RWMutex
// env maps from an environment variable to its value.
env map[string]string
)
func readenv(key string) (string, error) {
fd, err := Open("/env/"+key, O_RDONLY)
if err != nil {
return "", err
} }
f, e := Open("/env/"+key, O_RDONLY) defer Close(fd)
if e != nil { l, _ := Seek(fd, 0, 2)
return "", false Seek(fd, 0, 0)
buf := make([]byte, l)
n, err := Read(fd, buf)
if err != nil {
return "", err
} }
defer Close(f) if n > 0 && buf[n-1] == 0 {
buf = buf[:n-1]
}
return string(buf), nil
}
l, _ := Seek(f, 0, 2) func writeenv(key, value string) error {
Seek(f, 0, 0) fd, err := Create("/env/"+key, O_RDWR, 0666)
buf := make([]byte, l) if err != nil {
n, e := Read(f, buf) return err
if e != nil { }
defer Close(fd)
_, err = Write(fd, []byte(value))
return err
}
func copyenv() {
env = make(map[string]string)
fd, err := Open("/env", O_RDONLY)
if err != nil {
return
}
defer Close(fd)
files, err := readdirnames(fd)
if err != nil {
return
}
for _, key := range files {
v, err := readenv(key)
if err != nil {
continue
}
env[key] = v
}
}
func Getenv(key string) (value string, found bool) {
envOnce.Do(copyenv)
if len(key) == 0 {
return "", false return "", false
} }
if n > 0 && buf[n-1] == 0 { envLock.RLock()
buf = buf[:n-1] defer envLock.RUnlock()
v, ok := env[key]
if !ok {
return "", false
} }
return string(buf), true return v, true
} }
func Setenv(key, value string) error { func Setenv(key, value string) error {
envOnce.Do(copyenv)
if len(key) == 0 { if len(key) == 0 {
return errors.New("bad arg in system call") return errors.New("zero length key")
} }
f, e := Create("/env/"+key, O_RDWR, 0666) envLock.Lock()
if e != nil { defer envLock.Unlock()
return e
}
defer Close(f)
_, e = Write(f, []byte(value)) err := writeenv(key, value)
if err != nil {
return err
}
env[key] = value
return nil return nil
} }
func Clearenv() { func Clearenv() {
envOnce.Do(copyenv) // prevent copyenv in Getenv/Setenv
envLock.Lock()
defer envLock.Unlock()
env = make(map[string]string)
RawSyscall(SYS_RFORK, RFCENVG, 0, 0) RawSyscall(SYS_RFORK, RFCENVG, 0, 0)
} }
func Environ() []string { func Environ() []string {
env := make([]string, 0, 100) envOnce.Do(copyenv)
envLock.RLock()
f, e := Open("/env", O_RDONLY) defer envLock.RUnlock()
if e != nil { a := make([]string, len(env))
panic(e) i := 0
} for k, v := range env {
defer Close(f) a[i] = k + "=" + v
i++
names, e := readdirnames(f)
if e != nil {
panic(e)
}
for _, k := range names {
if v, ok := Getenv(k); ok {
env = append(env, k+"="+v)
}
} }
return env[0:len(env)] return a
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment