146 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			146 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // +build linux
 | |
| 
 | |
| /*
 | |
|    Copyright The containerd 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 mount
 | |
| 
 | |
| import (
 | |
| 	"bufio"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"os"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/pkg/errors"
 | |
| )
 | |
| 
 | |
| // Self retrieves a list of mounts for the current running process.
 | |
| func Self() ([]Info, error) {
 | |
| 	f, err := os.Open("/proc/self/mountinfo")
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	defer f.Close()
 | |
| 
 | |
| 	return parseInfoFile(f)
 | |
| }
 | |
| 
 | |
| func parseInfoFile(r io.Reader) ([]Info, error) {
 | |
| 	s := bufio.NewScanner(r)
 | |
| 	out := []Info{}
 | |
| 	var err error
 | |
| 	for s.Scan() {
 | |
| 		/*
 | |
| 		   See http://man7.org/linux/man-pages/man5/proc.5.html
 | |
| 
 | |
| 		   36 35 98:0 /mnt1 /mnt2 rw,noatime master:1 - ext3 /dev/root rw,errors=continue
 | |
| 		   (1)(2)(3)   (4)   (5)      (6)      (7)   (8) (9)   (10)         (11)
 | |
| 		   (1) mount ID:  unique identifier of the mount (may be reused after umount)
 | |
| 		   (2) parent ID:  ID of parent (or of self for the top of the mount tree)
 | |
| 		   (3) major:minor:  value of st_dev for files on filesystem
 | |
| 		   (4) root:  root of the mount within the filesystem
 | |
| 		   (5) mount point:  mount point relative to the process's root
 | |
| 		   (6) mount options:  per mount options
 | |
| 		   (7) optional fields:  zero or more fields of the form "tag[:value]"
 | |
| 		   (8) separator:  marks the end of the optional fields
 | |
| 		   (9) filesystem type:  name of filesystem of the form "type[.subtype]"
 | |
| 		   (10) mount source:  filesystem specific information or "none"
 | |
| 		   (11) super options:  per super block options
 | |
| 		*/
 | |
| 
 | |
| 		text := s.Text()
 | |
| 		fields := strings.Split(text, " ")
 | |
| 		numFields := len(fields)
 | |
| 		if numFields < 10 {
 | |
| 			// should be at least 10 fields
 | |
| 			return nil, errors.Errorf("parsing '%s' failed: not enough fields (%d)", text, numFields)
 | |
| 		}
 | |
| 		p := Info{}
 | |
| 		// ignore any numbers parsing errors, as there should not be any
 | |
| 		p.ID, _ = strconv.Atoi(fields[0])
 | |
| 		p.Parent, _ = strconv.Atoi(fields[1])
 | |
| 		mm := strings.Split(fields[2], ":")
 | |
| 		if len(mm) != 2 {
 | |
| 			return nil, errors.Errorf("parsing '%s' failed: unexpected minor:major pair %s", text, mm)
 | |
| 		}
 | |
| 		p.Major, _ = strconv.Atoi(mm[0])
 | |
| 		p.Minor, _ = strconv.Atoi(mm[1])
 | |
| 
 | |
| 		p.Root, err = strconv.Unquote(`"` + strings.Replace(fields[3], `"`, `\"`, -1) + `"`)
 | |
| 		if err != nil {
 | |
| 			return nil, errors.Wrapf(err, "parsing '%s' failed: unable to unquote root field", fields[3])
 | |
| 		}
 | |
| 		p.Mountpoint, err = strconv.Unquote(`"` + strings.Replace(fields[4], `"`, `\"`, -1) + `"`)
 | |
| 		if err != nil {
 | |
| 			return nil, errors.Wrapf(err, "parsing '%s' failed: unable to unquote mount point field", fields[4])
 | |
| 		}
 | |
| 		p.Options = fields[5]
 | |
| 
 | |
| 		// one or more optional fields, when a separator (-)
 | |
| 		i := 6
 | |
| 		for ; i < numFields && fields[i] != "-"; i++ {
 | |
| 			switch i {
 | |
| 			case 6:
 | |
| 				p.Optional = fields[6]
 | |
| 			default:
 | |
| 				/* NOTE there might be more optional fields before the separator
 | |
| 				   such as fields[7]...fields[N] (where N < separatorIndex),
 | |
| 				   although as of Linux kernel 4.15 the only known ones are
 | |
| 				   mount propagation flags in fields[6]. The correct
 | |
| 				   behavior is to ignore any unknown optional fields.
 | |
| 				*/
 | |
| 			}
 | |
| 		}
 | |
| 		if i == numFields {
 | |
| 			return nil, errors.Errorf("parsing '%s' failed: missing separator ('-')", text)
 | |
| 		}
 | |
| 		// There should be 3 fields after the separator...
 | |
| 		if i+4 > numFields {
 | |
| 			return nil, errors.Errorf("parsing '%s' failed: not enough fields after a separator", text)
 | |
| 		}
 | |
| 		// ... but in Linux <= 3.9 mounting a cifs with spaces in a share name
 | |
| 		// (like "//serv/My Documents") _may_ end up having a space in the last field
 | |
| 		// of mountinfo (like "unc=//serv/My Documents"). Since kernel 3.10-rc1, cifs
 | |
| 		// option unc= is ignored,  so a space should not appear. In here we ignore
 | |
| 		// those "extra" fields caused by extra spaces.
 | |
| 		p.FSType = fields[i+1]
 | |
| 		p.Source = fields[i+2]
 | |
| 		p.VFSOptions = fields[i+3]
 | |
| 
 | |
| 		out = append(out, p)
 | |
| 	}
 | |
| 	if err = s.Err(); err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	return out, nil
 | |
| }
 | |
| 
 | |
| // PID collects the mounts for a specific process ID. If the process
 | |
| // ID is unknown, it is better to use `Self` which will inspect
 | |
| // "/proc/self/mountinfo" instead.
 | |
| func PID(pid int) ([]Info, error) {
 | |
| 	f, err := os.Open(fmt.Sprintf("/proc/%d/mountinfo", pid))
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	defer f.Close()
 | |
| 
 | |
| 	return parseInfoFile(f)
 | |
| }
 | 
