Mike Frysinger | ae30210 | 2007-02-14 13:20:29 +0000 | [diff] [blame] | 1 | ------------- |
| 2 | MDEV Primer |
| 3 | ------------- |
| 4 | |
| 5 | For those of us who know how to use mdev, a primer might seem lame. For |
| 6 | everyone else, mdev is a weird black box that they hear is awesome, but can't |
| 7 | seem to get their head around how it works. Thus, a primer. |
| 8 | |
| 9 | ----------- |
| 10 | Basic Use |
| 11 | ----------- |
| 12 | |
| 13 | Mdev has two primary uses: initial population and dynamic updates. Both |
| 14 | require sysfs support in the kernel and have it mounted at /sys. For dynamic |
| 15 | updates, you also need to have hotplugging enabled in your kernel. |
| 16 | |
| 17 | Here's a typical code snippet from the init script: |
Bernhard Reutner-Fischer | 9c3052a | 2008-05-16 11:36:32 +0000 | [diff] [blame] | 18 | [0] mount -t proc proc /proc |
Mike Frysinger | ae30210 | 2007-02-14 13:20:29 +0000 | [diff] [blame] | 19 | [1] mount -t sysfs sysfs /sys |
| 20 | [2] echo /bin/mdev > /proc/sys/kernel/hotplug |
| 21 | [3] mdev -s |
| 22 | |
Bernhard Reutner-Fischer | 9c3052a | 2008-05-16 11:36:32 +0000 | [diff] [blame] | 23 | Alternatively, without procfs the above becomes: |
| 24 | [1] mount -t sysfs sysfs /sys |
| 25 | [2] sysctl -w kernel.hotplug=/bin/mdev |
| 26 | [3] mdev -s |
| 27 | |
| 28 | |
Mike Frysinger | ae30210 | 2007-02-14 13:20:29 +0000 | [diff] [blame] | 29 | Of course, a more "full" setup would entail executing this before the previous |
| 30 | code snippet: |
Bernhard Reutner-Fischer | 9c3052a | 2008-05-16 11:36:32 +0000 | [diff] [blame] | 31 | [4] mount -t tmpfs -o size=64k,mode=0755 tmpfs /dev |
Mike Frysinger | ae30210 | 2007-02-14 13:20:29 +0000 | [diff] [blame] | 32 | [5] mkdir /dev/pts |
| 33 | [6] mount -t devpts devpts /dev/pts |
| 34 | |
| 35 | The simple explanation here is that [1] you need to have /sys mounted before |
| 36 | executing mdev. Then you [2] instruct the kernel to execute /bin/mdev whenever |
| 37 | a device is added or removed so that the device node can be created or |
| 38 | destroyed. Then you [3] seed /dev with all the device nodes that were created |
| 39 | while the system was booting. |
| 40 | |
| 41 | For the "full" setup, you want to [4] make sure /dev is a tmpfs filesystem |
| 42 | (assuming you're running out of flash). Then you want to [5] create the |
| 43 | /dev/pts mount point and finally [6] mount the devpts filesystem on it. |
| 44 | |
| 45 | ------------- |
| 46 | MDEV Config (/etc/mdev.conf) |
| 47 | ------------- |
| 48 | |
| 49 | Mdev has an optional config file for controlling ownership/permissions of |
| 50 | device nodes if your system needs something more than the default root/root |
| 51 | 660 permissions. |
| 52 | |
| 53 | The file has the format: |
| 54 | <device regex> <uid>:<gid> <octal permissions> |
| 55 | For example: |
| 56 | hd[a-z][0-9]* 0:3 660 |
| 57 | |
| 58 | The config file parsing stops at the first matching line. If no line is |
| 59 | matched, then the default of 0:0 660 is used. To set your own default, simply |
| 60 | create your own total match like so: |
| 61 | .* 1:1 777 |
| 62 | |
Denis Vlasenko | df96df9 | 2008-07-26 18:35:10 +0000 | [diff] [blame] | 63 | You can rename/move device nodes by using the next optional field. |
| 64 | <device regex> <uid>:<gid> <octal permissions> [=path] |
Mike Frysinger | f0044c4 | 2008-02-01 06:53:50 +0000 | [diff] [blame] | 65 | So if you want to place the device node into a subdirectory, make sure the path |
| 66 | has a trailing /. If you want to rename the device node, just place the name. |
Denis Vlasenko | df96df9 | 2008-07-26 18:35:10 +0000 | [diff] [blame] | 67 | hda 0:3 660 =drives/ |
| 68 | This will move "hda" into the drives/ subdirectory. |
| 69 | hdb 0:3 660 =cdrom |
Mike Frysinger | f0044c4 | 2008-02-01 06:53:50 +0000 | [diff] [blame] | 70 | This will rename "hdb" to "cdrom". |
| 71 | |
Denis Vlasenko | df96df9 | 2008-07-26 18:35:10 +0000 | [diff] [blame] | 72 | Similarly, ">path" renames/moves the device but it also creates |
| 73 | a direct symlink /dev/DEVNAME to the renamed/moved device. |
| 74 | |
Mike Frysinger | ae30210 | 2007-02-14 13:20:29 +0000 | [diff] [blame] | 75 | If you also enable support for executing your own commands, then the file has |
| 76 | the format: |
Denis Vlasenko | df96df9 | 2008-07-26 18:35:10 +0000 | [diff] [blame] | 77 | <device regex> <uid>:<gid> <octal permissions> [=path] [@|$|*<command>] |
| 78 | or |
| 79 | <device regex> <uid>:<gid> <octal permissions> [>path] [@|$|*<command>] |
Mike Frysinger | ae30210 | 2007-02-14 13:20:29 +0000 | [diff] [blame] | 80 | The special characters have the meaning: |
| 81 | @ Run after creating the device. |
| 82 | $ Run before removing the device. |
| 83 | * Run both after creating and before removing the device. |
| 84 | |
| 85 | The command is executed via the system() function (which means you're giving a |
Mike Frysinger | c348e0b | 2008-02-01 01:41:57 +0000 | [diff] [blame] | 86 | command to the shell), so make sure you have a shell installed at /bin/sh. You |
| 87 | should also keep in mind that the kernel executes hotplug helpers with stdin, |
| 88 | stdout, and stderr connected to /dev/null. |
Mike Frysinger | ae30210 | 2007-02-14 13:20:29 +0000 | [diff] [blame] | 89 | |
| 90 | For your convenience, the shell env var $MDEV is set to the device name. So if |
Mike Frysinger | c348e0b | 2008-02-01 01:41:57 +0000 | [diff] [blame] | 91 | the device "hdc" was matched, MDEV would be set to "hdc". |
Mike Frysinger | ae7f7eb | 2007-06-28 17:13:51 +0000 | [diff] [blame] | 92 | |
| 93 | ---------- |
| 94 | FIRMWARE |
| 95 | ---------- |
| 96 | |
| 97 | Some kernel device drivers need to request firmware at runtime in order to |
| 98 | properly initialize a device. Place all such firmware files into the |
| 99 | /lib/firmware/ directory. At runtime, the kernel will invoke mdev with the |
| 100 | filename of the firmware which mdev will load out of /lib/firmware/ and into |
| 101 | the kernel via the sysfs interface. The exact filename is hardcoded in the |
Bernhard Reutner-Fischer | 9c3052a | 2008-05-16 11:36:32 +0000 | [diff] [blame] | 102 | kernel, so look there if you need to know how to name the file in userspace. |
Denis Vlasenko | 018bee6 | 2008-07-15 22:33:13 +0000 | [diff] [blame] | 103 | |
| 104 | ------------ |
| 105 | SEQUENCING |
| 106 | ------------ |
| 107 | |
| 108 | Kernel does not serialize hotplug events. It increments SEQNUM environmental |
| 109 | variable for each successive hotplug invocation. Normally, mdev doesn't care. |
| 110 | This may reorder hotplug and hot-unplug events, with typical symptoms of |
| 111 | device nodes sometimes not created as expected. |
| 112 | |
| 113 | However, if /dev/mdev.seq file is found, mdev will compare its |
| 114 | contents with SEQNUM. It will retry up to two seconds, waiting for them |
| 115 | to match. If they match exactly (not even trailing '\n' is allowed), |
| 116 | or if two seconds pass, mdev runs as usual, then it rewrites /dev/mdev.seq |
| 117 | with SEQNUM+1. |
| 118 | |
| 119 | IOW: this will serialize concurrent mdev invocations. |
| 120 | |
| 121 | If you want to activate this feature, execute "echo >/dev/mdev.seq" prior to |
| 122 | setting mdev to be the hotplug handler. This writes single '\n' to the file. |
| 123 | NB: mdev recognizes /dev/mdev.seq consisting of single '\n' characher |
| 124 | as a special case. IOW: this will not make your first hotplug event |
| 125 | to stall for two seconds. |