1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147
|
// -*- Mode: Go; indent-tabs-mode: t -*-
/*
* Copyright (C) 2017 Canonical Ltd
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 3 as
* published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
package kmod
import (
"sort"
"strings"
"github.com/snapcore/snapd/interfaces"
"github.com/snapcore/snapd/snap"
)
// Specification assists in collecting kernel modules associated with an interface.
//
// Unlike the Backend itself (which is stateless and non-persistent) this type
// holds internal state that is used by the kmod backend during the interface
// setup process.
type Specification struct {
modules map[string]bool
moduleOptions map[string]string
disallowedModules map[string]bool
}
// AddModule adds a kernel module, trimming spaces and ignoring duplicated modules.
func (spec *Specification) AddModule(module string) error {
m := strings.TrimSpace(module)
if m == "" {
return nil
}
if spec.modules == nil {
spec.modules = make(map[string]bool)
}
spec.modules[m] = true
return nil
}
// Modules returns a copy of the kernel module names added.
func (spec *Specification) Modules() map[string]bool {
result := make(map[string]bool, len(spec.modules))
for k, v := range spec.modules {
result[k] = v
}
return result
}
// SetModuleOptions specifies which options to use when loading the given kernel module.
func (spec *Specification) SetModuleOptions(module, options string) error {
if spec.moduleOptions == nil {
spec.moduleOptions = make(map[string]string)
}
spec.moduleOptions[module] = options
return nil
}
// moduleOptions returns the load options for each kernel module
func (spec *Specification) ModuleOptions() map[string]string {
return spec.moduleOptions
}
// DisallowModule adds a kernel module to the list of disallowed modules.
func (spec *Specification) DisallowModule(module string) error {
m := strings.TrimSpace(module)
if m == "" {
return nil
}
if spec.disallowedModules == nil {
spec.disallowedModules = make(map[string]bool)
}
spec.disallowedModules[m] = true
return nil
}
// DisallowedModules returns the list of disallowed modules.
func (spec *Specification) DisallowedModules() []string {
result := make([]string, 0, len(spec.disallowedModules))
for k, v := range spec.disallowedModules {
if v {
result = append(result, k)
}
}
sort.Strings(result)
return result
}
// Implementation of methods required by interfaces.Specification
// AddConnectedPlug records kmod-specific side-effects of having a connected plug.
func (spec *Specification) AddConnectedPlug(iface interfaces.Interface, plug *interfaces.ConnectedPlug, slot *interfaces.ConnectedSlot) error {
type definer interface {
KModConnectedPlug(spec *Specification, plug *interfaces.ConnectedPlug, slot *interfaces.ConnectedSlot) error
}
if iface, ok := iface.(definer); ok {
return iface.KModConnectedPlug(spec, plug, slot)
}
return nil
}
// AddConnectedSlot records kmod-specific side-effects of having a connected slot.
func (spec *Specification) AddConnectedSlot(iface interfaces.Interface, plug *interfaces.ConnectedPlug, slot *interfaces.ConnectedSlot) error {
type definer interface {
KModConnectedSlot(spec *Specification, plug *interfaces.ConnectedPlug, slot *interfaces.ConnectedSlot) error
}
if iface, ok := iface.(definer); ok {
return iface.KModConnectedSlot(spec, plug, slot)
}
return nil
}
// AddPermanentPlug records kmod-specific side-effects of having a plug.
func (spec *Specification) AddPermanentPlug(iface interfaces.Interface, plug *snap.PlugInfo) error {
type definer interface {
KModPermanentPlug(spec *Specification, plug *snap.PlugInfo) error
}
if iface, ok := iface.(definer); ok {
return iface.KModPermanentPlug(spec, plug)
}
return nil
}
// AddPermanentSlot records kmod-specific side-effects of having a slot.
func (spec *Specification) AddPermanentSlot(iface interfaces.Interface, slot *snap.SlotInfo) error {
type definer interface {
KModPermanentSlot(spec *Specification, slot *snap.SlotInfo) error
}
if iface, ok := iface.(definer); ok {
return iface.KModPermanentSlot(spec, slot)
}
return nil
}
|