mirror of
				https://github.com/optim-enterprises-bv/kubernetes.git
				synced 2025-10-31 02:08:13 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			302 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			302 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| //go:build linux
 | |
| // +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"
 | |
| 	"net"
 | |
| 	"os"
 | |
| 	"path/filepath"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 
 | |
| 	"k8s.io/klog/v2"
 | |
| )
 | |
| 
 | |
| // 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 underlying 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
 | |
| 				// JoinHostPort will add `[]` around IPv6 addresses.
 | |
| 				portal := net.JoinHostPort(strings.TrimSpace(string(addr)), strings.TrimSpace(string(port)))
 | |
| 				portalHostMap[portal] = hostNumber
 | |
| 
 | |
| 				persistentPortal := net.JoinHostPort(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
 | |
| }
 | 
