forked from mendersoftware/mender
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathioctl.go
158 lines (127 loc) · 3.88 KB
/
ioctl.go
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
148
149
150
151
152
153
154
155
156
157
158
// Copyright 2017 Northern.tech AS
//
// 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 main
import (
"errors"
"os"
"strings"
"syscall"
"unsafe"
"github.com/ungerik/go-sysfs"
)
// This is a bit weird, Syscall() says it accepts uintptr in the request field,
// but this in fact not true. By inspecting the calls with strace, it's clear
// that the pointer value is being passed as an int to ioctl(), which is just
// wrong. So write the ioctl request value (int) directly into the pointer value
// instead.
type ioctlRequestValue uintptr
var NotABlockDevice = errors.New("Not a block device.")
func isUbiBlockDevice(deviceName string) bool {
return sysfs.Class.Object("ubi").SubObject(deviceName).Exists()
}
func setUbiUpdateVolume(file *os.File, imageSize int64) error {
err := ioctlWrite(file.Fd(), UBI_IOCVOLUP, imageSize)
if err != nil {
return err
}
return nil
}
func getUbiDeviceSectorSize(file *os.File) (int, error) {
dev := strings.TrimPrefix(file.Name(), "/dev/")
ebSize := sysfs.Class.Object("ubi").SubObject(dev).Attribute("usable_eb_size")
if !ebSize.Exists() {
return 0, NotABlockDevice
}
sectorSize, err := ebSize.ReadUint64()
if err != nil {
return 0, NotABlockDevice
}
return int(sectorSize), nil
}
func getUbiDeviceSize(file *os.File) (uint64, error) {
dev := strings.TrimPrefix(file.Name(), "/dev/")
dataBytes := sysfs.Class.Object("ubi").SubObject(dev).Attribute("data_bytes")
if !dataBytes.Exists() {
return 0, NotABlockDevice
}
devSize, err := dataBytes.ReadUint64()
if err != nil {
return 0, NotABlockDevice
}
return devSize, nil
}
// Returns value in first return. Second returns error condition.
// If the device is not a block device NotABlockDevice error and
// value 0 will be returned.
func ioctlRead(fd uintptr, request ioctlRequestValue) (uint64, error) {
var response uint64
_, _, errno := syscall.Syscall(syscall.SYS_IOCTL, fd,
uintptr(unsafe.Pointer(request)),
uintptr(unsafe.Pointer(&response)))
if errno == syscall.ENOTTY {
// This means the descriptor is not a block device.
// ENOTTY... weird, I know.
return 0, NotABlockDevice
} else if errno != 0 {
return 0, errno
}
return response, nil
}
func ioctlWrite(fd uintptr, request ioctlRequestValue, data int64) error {
_, _, errno := syscall.Syscall(syscall.SYS_IOCTL, fd,
uintptr(unsafe.Pointer(request)),
uintptr(unsafe.Pointer(&data)))
if errno == syscall.ENOTTY {
// This means the descriptor is not a block device.
// ENOTTY... weird, I know.
return NotABlockDevice
} else if errno != 0 {
return errno
}
return nil
}
func getBlockDeviceSectorSize(file *os.File) (int, error) {
var sectorSize int
blockSectorSize, err := ioctlRead(file.Fd(), BLKSSZGET)
if err != nil && err != NotABlockDevice {
return 0, err
}
if err == NotABlockDevice {
// Check if it is an UBI block device
sectorSize, err = getUbiDeviceSectorSize(file)
if err != nil {
return 0, err
}
} else {
sectorSize = int(blockSectorSize)
}
return sectorSize, nil
}
func getBlockDeviceSize(file *os.File) (uint64, error) {
var devSize uint64
blkSize, err := ioctlRead(file.Fd(), BLKGETSIZE64)
if err != nil && err != NotABlockDevice {
return 0, err
}
if err == NotABlockDevice {
// Check if it is an UBI block device
devSize, err = getUbiDeviceSize(file)
if err != nil {
return 0, err
}
} else {
devSize = blkSize
}
return devSize, nil
}