1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108
|
// Copyright 2018 The gVisor 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 kvm
import (
"bufio"
"fmt"
"io"
"os"
"regexp"
"strconv"
"gvisor.dev/gvisor/pkg/hostarch"
)
type virtualRegion struct {
region
accessType hostarch.AccessType
shared bool
offset uintptr
filename string
}
// mapsLine matches a single line from /proc/PID/maps.
var mapsLine = regexp.MustCompile("([0-9a-f]+)-([0-9a-f]+) ([r-][w-][x-][sp]) ([0-9a-f]+) [0-9a-f]{2,3}:[0-9a-f]{2,} [0-9]+\\s+(.*)")
// excludeRegion returns true if these regions should be excluded from the
// physical map. Virtual regions need to be excluded if get_user_pages will
// fail on those addresses, preventing KVM from satisfying EPT faults.
//
// This is called by the physical map functions, not applyVirtualRegions.
func excludeVirtualRegion(r virtualRegion) bool {
return false
}
// applyVirtualRegions parses the process maps file.
//
// Unlike mappedRegions, these are not consistent over time.
func applyVirtualRegions(fn func(vr virtualRegion)) error {
// Open /proc/self/maps.
f, err := os.Open("/proc/self/maps")
if err != nil {
return err
}
defer f.Close()
// Parse all entries.
r := bufio.NewReader(f)
for {
b, err := r.ReadBytes('\n')
if b != nil && len(b) > 0 {
m := mapsLine.FindSubmatch(b)
if m == nil {
// This should not happen: kernel bug?
return fmt.Errorf("badly formed line: %v", string(b))
}
start, err := strconv.ParseUint(string(m[1]), 16, 64)
if err != nil {
return fmt.Errorf("bad start address: %v", string(b))
}
end, err := strconv.ParseUint(string(m[2]), 16, 64)
if err != nil {
return fmt.Errorf("bad end address: %v", string(b))
}
read := m[3][0] == 'r'
write := m[3][1] == 'w'
execute := m[3][2] == 'x'
shared := m[3][3] == 's'
offset, err := strconv.ParseUint(string(m[4]), 16, 64)
if err != nil {
return fmt.Errorf("bad offset: %v", string(b))
}
fn(virtualRegion{
region: region{
virtual: uintptr(start),
length: uintptr(end - start),
},
accessType: hostarch.AccessType{
Read: read,
Write: write,
Execute: execute,
},
shared: shared,
offset: uintptr(offset),
filename: string(m[5]),
})
}
if err != nil && err == io.EOF {
break
} else if err != nil {
return err
}
}
return nil
}
|