Skip to content
This repository has been archived by the owner on Oct 22, 2024. It is now read-only.

Commit

Permalink
support traditional file IO
Browse files Browse the repository at this point in the history
This may be useful for certain classes of applications.

Fixes: #991
  • Loading branch information
pohly committed Dec 6, 2021
1 parent 60e252f commit 6395500
Show file tree
Hide file tree
Showing 21 changed files with 344 additions and 48 deletions.
2 changes: 2 additions & 0 deletions Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -186,6 +186,8 @@ KUSTOMIZE += $(subst kubernetes-base,kubernetes-1.21,$(subst X.XX,1.22,$(KUSTOMI
KUSTOMIZE += deploy/common/pmem-storageclass-default.yaml=deploy/kustomize/storageclass
KUSTOMIZE += deploy/common/pmem-storageclass-ext4.yaml=deploy/kustomize/storageclass-ext4
KUSTOMIZE += deploy/common/pmem-storageclass-xfs.yaml=deploy/kustomize/storageclass-xfs
KUSTOMIZE += deploy/common/pmem-storageclass-ext4-fileio.yaml=deploy/kustomize/storageclass-ext4-fileio
KUSTOMIZE += deploy/common/pmem-storageclass-xfs-fileio.yaml=deploy/kustomize/storageclass-xfs-fileio
KUSTOMIZE += deploy/common/pmem-storageclass-late-binding.yaml=deploy/kustomize/storageclass-late-binding
KUSTOMIZE += deploy/operator/pmem-csi-operator.yaml=deploy/kustomize/operator

Expand Down
11 changes: 11 additions & 0 deletions deploy/common/pmem-storageclass-ext4-fileio.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
# Generated with "make kustomize", do not edit!

apiVersion: storage.k8s.io/v1
kind: StorageClass
metadata:
name: pmem-csi-sc-ext4-fileio
parameters:
csi.storage.k8s.io/fstype: ext4
eraseafter: "true"
usage: FileIO
provisioner: pmem-csi.intel.com
11 changes: 11 additions & 0 deletions deploy/common/pmem-storageclass-xfs-fileio.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
# Generated with "make kustomize", do not edit!

apiVersion: storage.k8s.io/v1
kind: StorageClass
metadata:
name: pmem-csi-sc-xfs-fileio
parameters:
csi.storage.k8s.io/fstype: xfs
eraseafter: "false"
usage: FileIO
provisioner: pmem-csi.intel.com
6 changes: 6 additions & 0 deletions deploy/kustomize/patches/usage-fileio.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
apiVersion: storage.k8s.io/v1
kind: StorageClass
metadata:
name: pmem-csi-sc
parameters:
usage: FileIO
7 changes: 7 additions & 0 deletions deploy/kustomize/storageclass-ext4-fileio/kustomization.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
nameSuffix: -fileio

bases:
- ../storageclass-ext4

patchesStrategicMerge:
- ../patches/usage-fileio.yaml
7 changes: 7 additions & 0 deletions deploy/kustomize/storageclass-xfs-fileio/kustomization.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
nameSuffix: -fileio

bases:
- ../storageclass-xfs

patchesStrategicMerge:
- ../patches/usage-fileio.yaml
25 changes: 24 additions & 1 deletion docs/install.md
Original file line number Diff line number Diff line change
Expand Up @@ -576,7 +576,30 @@ following custom parameters in a storage class:
|---|-------|--------|-------------|
|`eraseAfter`|Clear all data by overwriting with zeroes after use and before deleting the volume|Yes|`true` (default), `false`|
|`kataContainers`|Prepare volume for use with DAX in Kata Containers.|Yes|`false/0/f/FALSE` (default), `true/1/t/TRUE`|

|`usage`|Determine how a volume is going to be used.|Yes|`AppDirect` (default), `FileIO`|

By default, volumes are created for AppDirect enabled applications:
- The [namespace
mode](https://docs.pmem.io/ndctl-user-guide/concepts/nvdimm-namespaces) is
`fsdax`.
- Mount parameters include `-o dax` (= [`-o dax=always` on newer kernels](https://www.kernel.org/doc/Documentation/filesystems/dax.txt))
which ensures that all files are automatically opened in DAX mode, i.e.
reads and writes directly access the underlying PMEM.

This might not be ideal for traditional file IO because the page cache is
bypassed, which may affect performance, and because applications have to be
prepared to deal with partially written data sectors in case of crashes. When
the goal is to run traditional applications, then `usage=FileIO` may be better:
- In direct mode, the [namespace
mode](https://docs.pmem.io/ndctl-user-guide/concepts/nvdimm-namespaces) is
`sector`.
- In LVM mode, the namespace mode is `fsdax` because currently
PMEM-CSI doesn't support LVM on top of other namespaces.
- Mount parameters do not include `-o dax`.

`kataContainers` and `usage=FileIO` are mutually exclusive because the former
is about making AppDirect available in Kata Containers. The normal volume
passthrough can be used for `usage=FileIO`.

### Creating volumes

Expand Down
2 changes: 1 addition & 1 deletion pkg/pmem-csi-driver/controllerserver-node.go
Original file line number Diff line number Diff line change
Expand Up @@ -272,7 +272,7 @@ func (cs *nodeControllerServer) createVolumeInternal(ctx context.Context,
}
}()
}
actualSize, err := cs.dm.CreateDevice(ctx, volumeID, uint64(asked))
actualSize, err := cs.dm.CreateDevice(ctx, volumeID, uint64(asked), p.GetUsage())
if err != nil {
code := codes.Internal
if errors.Is(err, pmemerr.NotEnoughSpace) {
Expand Down
13 changes: 11 additions & 2 deletions pkg/pmem-csi-driver/nodeserver.go
Original file line number Diff line number Diff line change
Expand Up @@ -183,7 +183,9 @@ func (ns *nodeServer) NodePublishVolume(ctx context.Context, req *csi.NodePublis
return nil, err
}
srcPath = device.Path
mountFlags = append(mountFlags, daxMountFlag)
if v.GetUsage() == parameters.UsageAppDirect {
mountFlags = append(mountFlags, daxMountFlag)
}
} else {
// Validate parameters.
v, err := parameters.Parse(parameters.PersistentVolumeOrigin, req.GetVolumeContext())
Expand Down Expand Up @@ -527,6 +529,11 @@ func (ns *nodeServer) NodeStageVolume(ctx context.Context, req *csi.NodeStageVol
requestedFsType = defaultFilesystem
}

v, err := parameters.Parse(parameters.PersistentVolumeOrigin, req.GetVolumeContext())
if err != nil {
return nil, status.Error(codes.InvalidArgument, "persistent volume context: "+err.Error())
}

// Serialize by VolumeId
volumeMutex.LockKey(req.GetVolumeId())
defer func() {
Expand Down Expand Up @@ -572,7 +579,9 @@ func (ns *nodeServer) NodeStageVolume(ctx context.Context, req *csi.NodeStageVol
}
}

mountOptions = append(mountOptions, daxMountFlag)
if v.GetUsage() == parameters.UsageAppDirect {
mountOptions = append(mountOptions, daxMountFlag)
}

if err = ns.mount(ctx, device.Path, stagingtargetPath, mountOptions, false /* raw block */); err != nil {
return nil, status.Error(codes.Internal, err.Error())
Expand Down
34 changes: 34 additions & 0 deletions pkg/pmem-csi-driver/parameters/parameters.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ import (

type Persistency string
type Origin int
type Usage string

// Beware of API and backwards-compatibility breaking when changing these string constants!
const (
Expand All @@ -27,6 +28,11 @@ const (
Size = "size"
DeviceMode = "deviceMode"

// Added in PMEM-CSI 1.1.0.
UsageModel = "usage"
UsageAppDirect Usage = "AppDirect"
UsageFileIO Usage = "FileIO"

// Kubernetes v1.16+ adds this key to NodePublishRequest.VolumeContext
// while provisioning ephemeral volume.
Ephemeral = "csi.storage.k8s.io/ephemeral"
Expand Down Expand Up @@ -58,13 +64,15 @@ var valid = map[Origin][]string{
CreateVolumeOrigin: []string{
EraseAfter,
KataContainers,
UsageModel,
PersistencyModel,
},

// Parameters from Kubernetes and users.
EphemeralVolumeOrigin: []string{
EraseAfter,
KataContainers,
UsageModel,
PodInfoPrefix,
Size,
},
Expand All @@ -78,6 +86,7 @@ var valid = map[Origin][]string{
EraseAfter,
KataContainers,
PersistencyModel,
UsageModel,

Name,
PodInfoPrefix,
Expand All @@ -89,6 +98,7 @@ var valid = map[Origin][]string{
NodeVolumeOrigin: []string{
EraseAfter,
KataContainers,
UsageModel,
Name,
PersistencyModel,
Size,
Expand All @@ -107,6 +117,7 @@ type Volume struct {
Persistency *Persistency
Size *int64
DeviceMode *api.DeviceMode
Usage *Usage
}

// VolumeContext represents the same settings as a string map.
Expand Down Expand Up @@ -160,6 +171,15 @@ func Parse(origin Origin, stringmap map[string]string) (Volume, error) {
return result, fmt.Errorf("parameter %q: failed to parse %q as boolean: %v", key, value, err)
}
result.KataContainers = &b
case UsageModel:
u := Usage(value)
switch u {
case UsageAppDirect, UsageFileIO:
result.Usage = &u
case "":
default:
return result, fmt.Errorf("parameter %q: unknown value: %s", key, value)
}
case Size:
quantity, err := resource.ParseQuantity(value)
if err != nil {
Expand Down Expand Up @@ -201,6 +221,10 @@ func Parse(origin Origin, stringmap map[string]string) (Volume, error) {
return result, fmt.Errorf("required parameter %q not specified", Size)
}

if result.GetKataContainers() && result.GetUsage() != UsageAppDirect {
return result, fmt.Errorf("Kata Container support and usage %q are mutually exclusive", result.GetUsage())
}

return result, nil
}

Expand Down Expand Up @@ -235,6 +259,9 @@ func (v Volume) ToContext() VolumeContext {
if v.DeviceMode != nil {
result[DeviceMode] = string(*v.DeviceMode)
}
if v.Usage != nil {
result[UsageModel] = string(*v.Usage)
}

return result
}
Expand Down Expand Up @@ -281,3 +308,10 @@ func (v Volume) GetDeviceMode() api.DeviceMode {

return ""
}

func (v Volume) GetUsage() Usage {
if v.Usage != nil {
return *v.Usage
}
return UsageAppDirect
}
41 changes: 41 additions & 0 deletions pkg/pmem-csi-driver/parameters/parameters_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,8 @@ func TestParameters(t *testing.T) {
normal := PersistencyNormal
gig := "1Gi"
gigNum := int64(1 * 1024 * 1024 * 1024)
appDirect := UsageAppDirect
fileIO := UsageFileIO

tests := []struct {
name string
Expand Down Expand Up @@ -69,6 +71,45 @@ func TestParameters(t *testing.T) {
err: "parameter \"foo\" invalid in this context",
},

// Usage values.
{
name: "invalid-usage",
origin: CreateVolumeOrigin,
stringmap: VolumeContext{
UsageModel: "Foo",
},
err: "parameter \"usage\": unknown value: Foo",
},
{
name: "invalid-kata-containers",
origin: CreateVolumeOrigin,
stringmap: VolumeContext{
UsageModel: "FileIO",
KataContainers: "true",
},
err: "Kata Container support and usage \"FileIO\" are mutually exclusive",
},
{
name: "valid-usage-app-direct",
origin: CreateVolumeOrigin,
stringmap: VolumeContext{
UsageModel: "AppDirect",
},
parameters: Volume{
Usage: &appDirect,
},
},
{
name: "valid-usage-file-io",
origin: CreateVolumeOrigin,
stringmap: VolumeContext{
UsageModel: "FileIO",
},
parameters: Volume{
Usage: &fileIO,
},
},

// Parse errors for size.
{
name: "invalid-size-suffix",
Expand Down
3 changes: 2 additions & 1 deletion pkg/pmem-device-manager/pmd-fake.go
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import (
"sync"

api "github.com/intel/pmem-csi/pkg/apis/pmemcsi/v1beta1"
"github.com/intel/pmem-csi/pkg/pmem-csi-driver/parameters"

pmemerr "github.com/intel/pmem-csi/pkg/errors"
)
Expand Down Expand Up @@ -66,7 +67,7 @@ func (dm *fakeDM) getCapacity() Capacity {
}
}

func (dm *fakeDM) CreateDevice(ctx context.Context, volumeId string, size uint64) (uint64, error) {
func (dm *fakeDM) CreateDevice(ctx context.Context, volumeId string, size uint64, usage parameters.Usage) (uint64, error) {
dm.mutex.Lock()
defer dm.mutex.Unlock()

Expand Down
3 changes: 2 additions & 1 deletion pkg/pmem-device-manager/pmd-lvm.go
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ import (
pmemlog "github.com/intel/pmem-csi/pkg/logger"
"github.com/intel/pmem-csi/pkg/ndctl"
pmemcommon "github.com/intel/pmem-csi/pkg/pmem-common"
"github.com/intel/pmem-csi/pkg/pmem-csi-driver/parameters"
)

const (
Expand Down Expand Up @@ -132,7 +133,7 @@ func (lvm *pmemLvm) GetCapacity(ctx context.Context) (capacity Capacity, err err
return capacity, nil
}

func (lvm *pmemLvm) CreateDevice(ctx context.Context, volumeId string, size uint64) (uint64, error) {
func (lvm *pmemLvm) CreateDevice(ctx context.Context, volumeId string, size uint64, usage parameters.Usage) (uint64, error) {
ctx, logger := pmemlog.WithName(ctx, "LVM-CreateDevice")

lvmMutex.Lock()
Expand Down
4 changes: 3 additions & 1 deletion pkg/pmem-device-manager/pmd-manager.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@ import (

api "github.com/intel/pmem-csi/pkg/apis/pmemcsi/v1beta1"
"k8s.io/apimachinery/pkg/api/resource"

"github.com/intel/pmem-csi/pkg/pmem-csi-driver/parameters"
)

const (
Expand Down Expand Up @@ -79,7 +81,7 @@ type PmemDeviceManager interface {
// CreateDevice creates a new block device with give name, size and namespace mode.
// It returns the actual volume size which will always be at least as large as requested.
// Possible errors: ErrNotEnoughSpace, ErrDeviceExists
CreateDevice(ctx context.Context, name string, size uint64) (uint64, error)
CreateDevice(ctx context.Context, name string, size uint64, usage parameters.Usage) (uint64, error)

// GetDevice returns the block device information for given name
// Possible errors: ErrDeviceNotFound
Expand Down
11 changes: 6 additions & 5 deletions pkg/pmem-device-manager/pmd-manager_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ import (

pmemerr "github.com/intel/pmem-csi/pkg/errors"
pmemexec "github.com/intel/pmem-csi/pkg/exec"
"github.com/intel/pmem-csi/pkg/pmem-csi-driver/parameters"
. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"

Expand Down Expand Up @@ -90,7 +91,7 @@ func runTests(mode string) {
It("Should create a new device", func() {
name := "test-dev-new"
size := uint64(2) * 1024 * 1024 // 2Mb
actual, err := dm.CreateDevice(ctx, name, size)
actual, err := dm.CreateDevice(ctx, name, size, parameters.UsageAppDirect)
Expect(err).Should(BeNil(), "Failed to create new device")
Expect(actual).Should(BeNumerically(">=", size), "device at least as large as requested")

Expand All @@ -106,7 +107,7 @@ func runTests(mode string) {
It("Should support recreating a device", func() {
name := "test-dev"
size := uint64(2) * 1024 * 1024 // 2Mb
actual, err := dm.CreateDevice(ctx, name, size)
actual, err := dm.CreateDevice(ctx, name, size, parameters.UsageAppDirect)
Expect(err).Should(BeNil(), "Failed to create new device")
Expect(actual).Should(BeNumerically(">=", size), "device at least as large as requested")

Expand All @@ -122,7 +123,7 @@ func runTests(mode string) {
Expect(err).Should(BeNil(), "Failed to delete device")
cleanupList[name] = false

actual, err = dm.CreateDevice(ctx, name, size)
actual, err = dm.CreateDevice(ctx, name, size, parameters.UsageAppDirect)
Expect(err).Should(BeNil(), "Failed to recreate the same device")
Expect(actual).Should(BeNumerically(">=", size), "device at least as large as requested")
cleanupList[name] = true
Expand Down Expand Up @@ -151,7 +152,7 @@ func runTests(mode string) {
for i := 1; i <= max_devices; i++ {
name := fmt.Sprintf("list-dev-%d", i)
sizes[name] = uint64(rand.Intn(15)+1) * 1024 * 1024
actual, err := dm.CreateDevice(ctx, name, sizes[name])
actual, err := dm.CreateDevice(ctx, name, sizes[name], parameters.UsageAppDirect)
Expect(err).Should(BeNil(), "Failed to create new device")
Expect(actual).Should(BeNumerically(">=", sizes[name]), "device at least as large as requested")
cleanupList[name] = true
Expand Down Expand Up @@ -195,7 +196,7 @@ func runTests(mode string) {
It("Should delete devices", func() {
name := "delete-dev"
size := uint64(2) * 1024 * 1024 // 2Mb
actual, err := dm.CreateDevice(ctx, name, size)
actual, err := dm.CreateDevice(ctx, name, size, parameters.UsageAppDirect)
Expect(err).Should(BeNil(), "Failed to create new device")
Expect(actual).Should(BeNumerically(">=", size), "device at least as large as requested")
cleanupList[name] = true
Expand Down
Loading

0 comments on commit 6395500

Please sign in to comment.