- Fix target delete URL path mismatch (/targets/ -> /target/) - Implement target create/delete server handlers with proper validation - Add DeleteTarget method with force flag and mutex locking to SCSITargetService - Implement full LU management: create/list/delete through CLI, client, and server - Add TPGT list command to show target portal group tags - Add unit tests for target/LU router handlers and SCSI service Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
186 lines
4.6 KiB
Go
186 lines
4.6 KiB
Go
/*
|
|
Copyright 2016 The GoStor Authors All rights reserved.
|
|
|
|
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 scsi
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"strconv"
|
|
"sync"
|
|
|
|
"github.com/gostor/gotgt/pkg/api"
|
|
"github.com/gostor/gotgt/pkg/config"
|
|
)
|
|
|
|
type BackendType string
|
|
|
|
type SCSILUMap struct {
|
|
mutex sync.RWMutex
|
|
// use UUID as the key for all LUs
|
|
AllDevices api.LUNMap
|
|
// use target name as the key for target's LUN map
|
|
TargetsLUNMap map[string]api.LUNMap
|
|
|
|
TargetsBSMap map[string]api.RemoteBackingStore /* use target name as the key for target's Backing Store (temp) */
|
|
}
|
|
|
|
var globalSCSILUMap = SCSILUMap{
|
|
AllDevices: make(api.LUNMap),
|
|
TargetsLUNMap: make(map[string]api.LUNMap),
|
|
TargetsBSMap: make(map[string]api.RemoteBackingStore),
|
|
}
|
|
|
|
type LUNMapping struct {
|
|
TargetName string
|
|
LUN uint64
|
|
DeviceID uint64
|
|
}
|
|
|
|
func mappingLUN(lm LUNMapping) {
|
|
device := globalSCSILUMap.AllDevices[lm.DeviceID]
|
|
lunMap := globalSCSILUMap.TargetsLUNMap[lm.TargetName]
|
|
if lunMap == nil {
|
|
globalSCSILUMap.TargetsLUNMap[lm.TargetName] = make(api.LUNMap)
|
|
lunMap = globalSCSILUMap.TargetsLUNMap[lm.TargetName]
|
|
}
|
|
lunMap[lm.LUN] = device
|
|
}
|
|
|
|
func GetLU(tgtName string, LUN uint64) *api.SCSILu {
|
|
globalSCSILUMap.mutex.RLock()
|
|
defer globalSCSILUMap.mutex.RUnlock()
|
|
|
|
lunMap := globalSCSILUMap.TargetsLUNMap[tgtName]
|
|
lun := lunMap[LUN]
|
|
|
|
return lun
|
|
}
|
|
|
|
func GetTargetLUNMap(tgtName string) api.LUNMap {
|
|
globalSCSILUMap.mutex.RLock()
|
|
defer globalSCSILUMap.mutex.RUnlock()
|
|
|
|
lunMap := globalSCSILUMap.TargetsLUNMap[tgtName]
|
|
return lunMap
|
|
}
|
|
|
|
func GetTargetBSMap(tgtName string) (api.RemoteBackingStore, error) {
|
|
globalSCSILUMap.mutex.RLock()
|
|
defer globalSCSILUMap.mutex.RUnlock()
|
|
|
|
bs, ok := globalSCSILUMap.TargetsBSMap[tgtName]
|
|
if !ok {
|
|
return nil, errors.New("Remote backing store is not found in globalSCSILUMap")
|
|
}
|
|
|
|
return bs, nil
|
|
}
|
|
|
|
func AddBackendStorage(bs config.BackendStorage) error {
|
|
globalSCSILUMap.mutex.Lock()
|
|
defer globalSCSILUMap.mutex.Unlock()
|
|
_, ok := globalSCSILUMap.AllDevices[bs.DeviceID]
|
|
if ok {
|
|
return fmt.Errorf("device %q already exists", bs.DeviceID)
|
|
}
|
|
|
|
lu, err := NewSCSILu(&bs)
|
|
if err != nil {
|
|
return fmt.Errorf("Init SCSI LU map error: %v", err)
|
|
}
|
|
globalSCSILUMap.AllDevices[bs.DeviceID] = lu
|
|
return nil
|
|
}
|
|
|
|
func DelBackendStorage(deviceID uint64) {
|
|
globalSCSILUMap.mutex.Lock()
|
|
defer globalSCSILUMap.mutex.Unlock()
|
|
delete(globalSCSILUMap.AllDevices, deviceID)
|
|
}
|
|
|
|
func AddLUNMapping(m LUNMapping) error {
|
|
globalSCSILUMap.mutex.Lock()
|
|
defer globalSCSILUMap.mutex.Unlock()
|
|
mappingLUN(m)
|
|
// Init SCSISimpleReservationOperator
|
|
op := GetSCSIReservationOperator()
|
|
if simpleOp, ok := op.(*SCSISimpleReservationOperator); ok {
|
|
simpleOp.InitLUReservation(m.TargetName, m.DeviceID)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func DelLUNMapping(m LUNMapping) {
|
|
globalSCSILUMap.mutex.Lock()
|
|
defer globalSCSILUMap.mutex.Unlock()
|
|
delete(globalSCSILUMap.TargetsLUNMap[m.TargetName], m.LUN)
|
|
}
|
|
|
|
// DelTargetLUNMap removes the entire LUN map for a target.
|
|
func DelTargetLUNMap(targetName string) {
|
|
globalSCSILUMap.mutex.Lock()
|
|
defer globalSCSILUMap.mutex.Unlock()
|
|
delete(globalSCSILUMap.TargetsLUNMap, targetName)
|
|
}
|
|
|
|
func InitSCSILUMap(config *config.Config) error {
|
|
for _, bs := range config.Storages {
|
|
if err := AddBackendStorage(bs); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
for tgtName, tgt := range config.ISCSITargets {
|
|
for lunstr, deviceID := range tgt.LUNs {
|
|
lun, err := strconv.ParseUint(lunstr, 10, 64)
|
|
if err != nil {
|
|
return fmt.Errorf("LU Number must be a number")
|
|
}
|
|
m := LUNMapping{DeviceID: deviceID, LUN: lun, TargetName: tgtName}
|
|
AddLUNMapping(m)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func InitSCSILUMapEx(config *config.BackendStorage, tgtName string, lun uint64, bs api.RemoteBackingStore) error {
|
|
if bs == nil {
|
|
return errors.New("Remote backing store is nil")
|
|
}
|
|
|
|
globalSCSILUMap.mutex.Lock()
|
|
globalSCSILUMap.TargetsBSMap[tgtName] = bs
|
|
globalSCSILUMap.mutex.Unlock()
|
|
|
|
lu, err := NewSCSILu(config)
|
|
if err != nil {
|
|
return fmt.Errorf("Init SCSI LU map error, err: %v", err)
|
|
}
|
|
|
|
globalSCSILUMap.mutex.Lock()
|
|
globalSCSILUMap.AllDevices[config.DeviceID] = lu
|
|
globalSCSILUMap.mutex.Unlock()
|
|
|
|
mappingLUN(LUNMapping{
|
|
DeviceID: config.DeviceID,
|
|
LUN: lun,
|
|
TargetName: tgtName,
|
|
},
|
|
)
|
|
return nil
|
|
}
|