123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300 |
- // +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/filepath"
- "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 := filepath.Join("/sys/block/", disk, "/slaves/")
- if files, err := io.ReadDir(slavesPath); err == nil {
- for _, f := range files {
- devices = append(devices, filepath.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
- }
|