mirror of https://github.com/k3s-io/k3s
300 lines
9.3 KiB
Go
300 lines
9.3 KiB
Go
// +build linux
|
|
|
|
/*
|
|
Copyright 2016 The Kubernetes 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 util
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"path"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"k8s.io/klog"
|
|
)
|
|
|
|
// FindMultipathDeviceForDevice given a device name like /dev/sdx, find the devicemapper parent
|
|
func (handler *deviceHandler) FindMultipathDeviceForDevice(device string) string {
|
|
io := handler.getIo
|
|
disk, err := findDeviceForPath(device, io)
|
|
if err != nil {
|
|
return ""
|
|
}
|
|
sysPath := "/sys/block/"
|
|
if dirs, err := io.ReadDir(sysPath); err == nil {
|
|
for _, f := range dirs {
|
|
name := f.Name()
|
|
if strings.HasPrefix(name, "dm-") {
|
|
if _, err1 := io.Lstat(sysPath + name + "/slaves/" + disk); err1 == nil {
|
|
return "/dev/" + name
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return ""
|
|
}
|
|
|
|
// findDeviceForPath Find the underlaying disk for a linked path such as /dev/disk/by-path/XXXX or /dev/mapper/XXXX
|
|
// will return sdX or hdX etc, if /dev/sdX is passed in then sdX will be returned
|
|
func findDeviceForPath(path string, io IoUtil) (string, error) {
|
|
devicePath, err := io.EvalSymlinks(path)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
// if path /dev/hdX split into "", "dev", "hdX" then we will
|
|
// return just the last part
|
|
parts := strings.Split(devicePath, "/")
|
|
if len(parts) == 3 && strings.HasPrefix(parts[1], "dev") {
|
|
return parts[2], nil
|
|
}
|
|
return "", errors.New("Illegal path for device " + devicePath)
|
|
}
|
|
|
|
// FindSlaveDevicesOnMultipath given a dm name like /dev/dm-1, find all devices
|
|
// which are managed by the devicemapper dm-1.
|
|
func (handler *deviceHandler) FindSlaveDevicesOnMultipath(dm string) []string {
|
|
var devices []string
|
|
io := handler.getIo
|
|
// Split path /dev/dm-1 into "", "dev", "dm-1"
|
|
parts := strings.Split(dm, "/")
|
|
if len(parts) != 3 || !strings.HasPrefix(parts[1], "dev") {
|
|
return devices
|
|
}
|
|
disk := parts[2]
|
|
slavesPath := path.Join("/sys/block/", disk, "/slaves/")
|
|
if files, err := io.ReadDir(slavesPath); err == nil {
|
|
for _, f := range files {
|
|
devices = append(devices, path.Join("/dev/", f.Name()))
|
|
}
|
|
}
|
|
return devices
|
|
}
|
|
|
|
// GetISCSIPortalHostMapForTarget given a target iqn, find all the scsi hosts logged into
|
|
// that target. Returns a map of iSCSI portals (string) to SCSI host numbers (integers).
|
|
// For example: {
|
|
// "192.168.30.7:3260": 2,
|
|
// "192.168.30.8:3260": 3,
|
|
// }
|
|
func (handler *deviceHandler) GetISCSIPortalHostMapForTarget(targetIqn string) (map[string]int, error) {
|
|
portalHostMap := make(map[string]int)
|
|
io := handler.getIo
|
|
|
|
// Iterate over all the iSCSI hosts in sysfs
|
|
sysPath := "/sys/class/iscsi_host"
|
|
hostDirs, err := io.ReadDir(sysPath)
|
|
if err != nil {
|
|
if os.IsNotExist(err) {
|
|
return portalHostMap, nil
|
|
}
|
|
return nil, err
|
|
}
|
|
for _, hostDir := range hostDirs {
|
|
// iSCSI hosts are always of the format "host%d"
|
|
// See drivers/scsi/hosts.c in Linux
|
|
hostName := hostDir.Name()
|
|
if !strings.HasPrefix(hostName, "host") {
|
|
continue
|
|
}
|
|
hostNumber, err := strconv.Atoi(strings.TrimPrefix(hostName, "host"))
|
|
if err != nil {
|
|
klog.Errorf("Could not get number from iSCSI host: %s", hostName)
|
|
continue
|
|
}
|
|
|
|
// Iterate over the children of the iscsi_host device
|
|
// We are looking for the associated session
|
|
devicePath := sysPath + "/" + hostName + "/device"
|
|
deviceDirs, err := io.ReadDir(devicePath)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, deviceDir := range deviceDirs {
|
|
// Skip over files that aren't the session
|
|
// Sessions are of the format "session%u"
|
|
// See drivers/scsi/scsi_transport_iscsi.c in Linux
|
|
sessionName := deviceDir.Name()
|
|
if !strings.HasPrefix(sessionName, "session") {
|
|
continue
|
|
}
|
|
|
|
sessionPath := devicePath + "/" + sessionName
|
|
|
|
// Read the target name for the iSCSI session
|
|
targetNamePath := sessionPath + "/iscsi_session/" + sessionName + "/targetname"
|
|
targetName, err := io.ReadFile(targetNamePath)
|
|
if err != nil {
|
|
klog.Infof("Failed to process session %s, assuming this session is unavailable: %s", sessionName, err)
|
|
continue
|
|
}
|
|
|
|
// Ignore hosts that don't matchthe target we were looking for.
|
|
if strings.TrimSpace(string(targetName)) != targetIqn {
|
|
continue
|
|
}
|
|
|
|
// Iterate over the children of the iSCSI session looking
|
|
// for the iSCSI connection.
|
|
dirs2, err := io.ReadDir(sessionPath)
|
|
if err != nil {
|
|
klog.Infof("Failed to process session %s, assuming this session is unavailable: %s", sessionName, err)
|
|
continue
|
|
}
|
|
for _, dir2 := range dirs2 {
|
|
// Skip over files that aren't the connection
|
|
// Connections are of the format "connection%d:%u"
|
|
// See drivers/scsi/scsi_transport_iscsi.c in Linux
|
|
dirName := dir2.Name()
|
|
if !strings.HasPrefix(dirName, "connection") {
|
|
continue
|
|
}
|
|
|
|
connectionPath := sessionPath + "/" + dirName + "/iscsi_connection/" + dirName
|
|
|
|
// Read the current and persistent portal information for the connection.
|
|
addrPath := connectionPath + "/address"
|
|
addr, err := io.ReadFile(addrPath)
|
|
if err != nil {
|
|
klog.Infof("Failed to process connection %s, assuming this connection is unavailable: %s", dirName, err)
|
|
continue
|
|
}
|
|
|
|
portPath := connectionPath + "/port"
|
|
port, err := io.ReadFile(portPath)
|
|
if err != nil {
|
|
klog.Infof("Failed to process connection %s, assuming this connection is unavailable: %s", dirName, err)
|
|
continue
|
|
}
|
|
|
|
persistentAddrPath := connectionPath + "/persistent_address"
|
|
persistentAddr, err := io.ReadFile(persistentAddrPath)
|
|
if err != nil {
|
|
klog.Infof("Failed to process connection %s, assuming this connection is unavailable: %s", dirName, err)
|
|
continue
|
|
}
|
|
|
|
persistentPortPath := connectionPath + "/persistent_port"
|
|
persistentPort, err := io.ReadFile(persistentPortPath)
|
|
if err != nil {
|
|
klog.Infof("Failed to process connection %s, assuming this connection is unavailable: %s", dirName, err)
|
|
continue
|
|
}
|
|
|
|
// Add entries to the map for both the current and persistent portals
|
|
// pointing to the SCSI host for those connections
|
|
portal := strings.TrimSpace(string(addr)) + ":" +
|
|
strings.TrimSpace(string(port))
|
|
portalHostMap[portal] = hostNumber
|
|
|
|
persistentPortal := strings.TrimSpace(string(persistentAddr)) + ":" +
|
|
strings.TrimSpace(string(persistentPort))
|
|
portalHostMap[persistentPortal] = hostNumber
|
|
}
|
|
}
|
|
}
|
|
|
|
return portalHostMap, nil
|
|
}
|
|
|
|
// FindDevicesForISCSILun given an iqn, and lun number, find all the devices
|
|
// corresponding to that LUN.
|
|
func (handler *deviceHandler) FindDevicesForISCSILun(targetIqn string, lun int) ([]string, error) {
|
|
devices := make([]string, 0)
|
|
io := handler.getIo
|
|
|
|
// Iterate over all the iSCSI hosts in sysfs
|
|
sysPath := "/sys/class/iscsi_host"
|
|
hostDirs, err := io.ReadDir(sysPath)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, hostDir := range hostDirs {
|
|
// iSCSI hosts are always of the format "host%d"
|
|
// See drivers/scsi/hosts.c in Linux
|
|
hostName := hostDir.Name()
|
|
if !strings.HasPrefix(hostName, "host") {
|
|
continue
|
|
}
|
|
hostNumber, err := strconv.Atoi(strings.TrimPrefix(hostName, "host"))
|
|
if err != nil {
|
|
klog.Errorf("Could not get number from iSCSI host: %s", hostName)
|
|
continue
|
|
}
|
|
|
|
// Iterate over the children of the iscsi_host device
|
|
// We are looking for the associated session
|
|
devicePath := sysPath + "/" + hostName + "/device"
|
|
deviceDirs, err := io.ReadDir(devicePath)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, deviceDir := range deviceDirs {
|
|
// Skip over files that aren't the session
|
|
// Sessions are of the format "session%u"
|
|
// See drivers/scsi/scsi_transport_iscsi.c in Linux
|
|
sessionName := deviceDir.Name()
|
|
if !strings.HasPrefix(sessionName, "session") {
|
|
continue
|
|
}
|
|
|
|
// Read the target name for the iSCSI session
|
|
targetNamePath := devicePath + "/" + sessionName + "/iscsi_session/" + sessionName + "/targetname"
|
|
targetName, err := io.ReadFile(targetNamePath)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Only if the session matches the target we were looking for,
|
|
// add it to the map
|
|
if strings.TrimSpace(string(targetName)) != targetIqn {
|
|
continue
|
|
}
|
|
|
|
// The list of block devices on the scsi bus will be in a
|
|
// directory called "target%d:%d:%d".
|
|
// See drivers/scsi/scsi_scan.c in Linux
|
|
// We assume the channel/bus and device/controller are always zero for iSCSI
|
|
targetPath := devicePath + "/" + sessionName + fmt.Sprintf("/target%d:0:0", hostNumber)
|
|
|
|
// The block device for a given lun will be "%d:%d:%d:%d" --
|
|
// host:channel:bus:LUN
|
|
blockDevicePath := targetPath + fmt.Sprintf("/%d:0:0:%d", hostNumber, lun)
|
|
|
|
// If the LUN doesn't exist on this bus, continue on
|
|
_, err = io.Lstat(blockDevicePath)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
|
|
// Read the block directory, there should only be one child --
|
|
// the block device "sd*"
|
|
path := blockDevicePath + "/block"
|
|
dirs, err := io.ReadDir(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if 0 < len(dirs) {
|
|
devices = append(devices, dirs[0].Name())
|
|
}
|
|
}
|
|
}
|
|
|
|
return devices, nil
|
|
}
|