4 This target emulates the behavior of bad sectors at arbitrary
5 locations, and the ability to enable the emulation of the failures
8 This target behaves similarly to a linear target. At a given time,
9 the user can send a message to the target to start failing read
10 requests on specific blocks (to emulate the behavior of a hard disk
11 drive with bad sectors).
13 When the failure behavior is enabled (i.e.: when the output of
14 "dmsetup status" displays "fail_read_on_bad_block"), reads of blocks
15 in the "bad block list" will fail with EIO ("Input/output error").
17 Writes of blocks in the "bad block list will result in the following:
19 1. Remove the block from the "bad block list".
20 2. Successfully complete the write.
22 This emulates the "remapped sector" behavior of a drive with bad
25 Normally, a drive that is encountering bad sectors will most likely
26 encounter more bad sectors, at an unknown time or location.
27 With dm-dust, the user can use the "addbadblock" and "removebadblock"
28 messages to add arbitrary bad blocks at new locations, and the
29 "enable" and "disable" messages to modulate the state of whether the
30 configured "bad blocks" will be treated as bad, or bypassed.
31 This allows the pre-writing of test data and metadata prior to
32 simulating a "failure" event where bad sectors start to appear.
36 <device_path> <offset> <blksz>
40 Path to the block device.
43 Offset to data area from start of device_path
48 (minimum 512, maximum 1073741824, must be a power of 2)
53 First, find the size (in 512-byte sectors) of the device to be used::
55 $ sudo blockdev --getsz /dev/vdb1
58 Create the dm-dust device:
59 (For a device with a block size of 512 bytes)
63 $ sudo dmsetup create dust1 --table '0 33552384 dust /dev/vdb1 0 512'
65 (For a device with a block size of 4096 bytes)
69 $ sudo dmsetup create dust1 --table '0 33552384 dust /dev/vdb1 0 4096'
71 Check the status of the read behavior ("bypass" indicates that all I/O
72 will be passed through to the underlying device)::
74 $ sudo dmsetup status dust1
75 0 33552384 dust 252:17 bypass
77 $ sudo dd if=/dev/mapper/dust1 of=/dev/null bs=512 count=128 iflag=direct
81 $ sudo dd if=/dev/zero of=/dev/mapper/dust1 bs=512 count=128 oflag=direct
85 Adding and removing bad blocks
86 ------------------------------
88 At any time (i.e.: whether the device has the "bad block" emulation
89 enabled or disabled), bad blocks may be added or removed from the
90 device via the "addbadblock" and "removebadblock" messages::
92 $ sudo dmsetup message dust1 0 addbadblock 60
93 kernel: device-mapper: dust: badblock added at block 60
95 $ sudo dmsetup message dust1 0 addbadblock 67
96 kernel: device-mapper: dust: badblock added at block 67
98 $ sudo dmsetup message dust1 0 addbadblock 72
99 kernel: device-mapper: dust: badblock added at block 72
101 These bad blocks will be stored in the "bad block list".
102 While the device is in "bypass" mode, reads and writes will succeed::
104 $ sudo dmsetup status dust1
105 0 33552384 dust 252:17 bypass
107 Enabling block read failures
108 ----------------------------
110 To enable the "fail read on bad block" behavior, send the "enable" message::
112 $ sudo dmsetup message dust1 0 enable
113 kernel: device-mapper: dust: enabling read failures on bad sectors
115 $ sudo dmsetup status dust1
116 0 33552384 dust 252:17 fail_read_on_bad_block
118 With the device in "fail read on bad block" mode, attempting to read a
119 block will encounter an "Input/output error"::
121 $ sudo dd if=/dev/mapper/dust1 of=/dev/null bs=512 count=1 skip=67 iflag=direct
122 dd: error reading '/dev/mapper/dust1': Input/output error
125 0 bytes copied, 0.00040651 s, 0.0 kB/s
127 ...and writing to the bad blocks will remove the blocks from the list,
128 therefore emulating the "remap" behavior of hard disk drives::
130 $ sudo dd if=/dev/zero of=/dev/mapper/dust1 bs=512 count=128 oflag=direct
134 kernel: device-mapper: dust: block 60 removed from badblocklist by write
135 kernel: device-mapper: dust: block 67 removed from badblocklist by write
136 kernel: device-mapper: dust: block 72 removed from badblocklist by write
137 kernel: device-mapper: dust: block 87 removed from badblocklist by write
139 Bad block add/remove error handling
140 -----------------------------------
142 Attempting to add a bad block that already exists in the list will
143 result in an "Invalid argument" error, as well as a helpful message::
145 $ sudo dmsetup message dust1 0 addbadblock 88
146 device-mapper: message ioctl on dust1 failed: Invalid argument
147 kernel: device-mapper: dust: block 88 already in badblocklist
149 Attempting to remove a bad block that doesn't exist in the list will
150 result in an "Invalid argument" error, as well as a helpful message::
152 $ sudo dmsetup message dust1 0 removebadblock 87
153 device-mapper: message ioctl on dust1 failed: Invalid argument
154 kernel: device-mapper: dust: block 87 not found in badblocklist
156 Counting the number of bad blocks in the bad block list
157 -------------------------------------------------------
159 To count the number of bad blocks configured in the device, run the
160 following message command::
162 $ sudo dmsetup message dust1 0 countbadblocks
164 A message will print with the number of bad blocks currently
165 configured on the device::
167 kernel: device-mapper: dust: countbadblocks: 895 badblock(s) found
169 Querying for specific bad blocks
170 --------------------------------
172 To find out if a specific block is in the bad block list, run the
173 following message command::
175 $ sudo dmsetup message dust1 0 queryblock 72
177 The following message will print if the block is in the list::
179 device-mapper: dust: queryblock: block 72 found in badblocklist
181 The following message will print if the block is not in the list::
183 device-mapper: dust: queryblock: block 72 not found in badblocklist
185 The "queryblock" message command will work in both the "enabled"
186 and "disabled" modes, allowing the verification of whether a block
187 will be treated as "bad" without having to issue I/O to the device,
188 or having to "enable" the bad block emulation.
190 Clearing the bad block list
191 ---------------------------
193 To clear the bad block list (without needing to individually run
194 a "removebadblock" message command for every block), run the
195 following message command::
197 $ sudo dmsetup message dust1 0 clearbadblocks
199 After clearing the bad block list, the following message will appear::
201 kernel: device-mapper: dust: clearbadblocks: badblocks cleared
203 If there were no bad blocks to clear, the following message will
206 kernel: device-mapper: dust: clearbadblocks: no badblocks found
208 Message commands list
209 ---------------------
211 Below is a list of the messages that can be sent to a dust device:
213 Operations on blocks (requires a <blknum> argument)::
217 removebadblock <blknum>
219 ...where <blknum> is a block number within range of the device
220 (corresponding to the block size of the device.)
222 Single argument message commands::
233 When finished, remove the device via the "dmsetup remove" command::
235 $ sudo dmsetup remove dust1
240 On test runs with many bad blocks, it may be desirable to avoid
241 excessive logging (from bad blocks added, removed, or "remapped").
242 This can be done by enabling "quiet mode" via the following message::
244 $ sudo dmsetup message dust1 0 quiet
246 This will suppress log messages from add / remove / removed by write
247 operations. Log messages from "countbadblocks" or "queryblock"
248 message commands will still print in quiet mode.
250 The status of quiet mode can be seen by running "dmsetup status"::
252 $ sudo dmsetup status dust1
253 0 33552384 dust 252:17 fail_read_on_bad_block quiet
255 To disable quiet mode, send the "quiet" message again::
257 $ sudo dmsetup message dust1 0 quiet
259 $ sudo dmsetup status dust1
260 0 33552384 dust 252:17 fail_read_on_bad_block verbose
262 (The presence of "verbose" indicates normal logging.)
267 scsi_debug has a "medium error" mode that can fail reads on one
268 specified sector (sector 0x1234, hardcoded in the source code), but
269 it uses RAM for the persistent storage, which drastically decreases
270 the potential device size.
272 dm-flakey fails all I/O from all block locations at a specified time
273 frequency, and not a given point in time.
275 When a bad sector occurs on a hard disk drive, reads to that sector
276 are failed by the device, usually resulting in an error code of EIO
277 ("I/O error") or ENODATA ("No data available"). However, a write to
278 the sector may succeed, and result in the sector becoming readable
279 after the device controller no longer experiences errors reading the
280 sector (or after a reallocation of the sector). However, there may
281 be bad sectors that occur on the device in the future, in a different,
282 unpredictable location.
284 This target seeks to provide a device that can exhibit the behavior
285 of a bad sector at a known sector location, at a known time, based
286 on a large storage device (at least tens of gigabytes, not occupying