ide-cd: document capacity hack
[deliverable/linux.git] / drivers / ide / ide-gd.c
CommitLineData
5fef0e5c
BZ
1#include <linux/module.h>
2#include <linux/types.h>
3#include <linux/string.h>
4#include <linux/kernel.h>
5#include <linux/errno.h>
6#include <linux/genhd.h>
7#include <linux/mutex.h>
8#include <linux/ide.h>
9#include <linux/hdreg.h>
10
11#if !defined(CONFIG_DEBUG_BLOCK_EXT_DEVT)
12#define IDE_DISK_MINORS (1 << PARTN_BITS)
13#else
14#define IDE_DISK_MINORS 0
15#endif
16
17#include "ide-disk.h"
806f80a6 18#include "ide-floppy.h"
5fef0e5c
BZ
19
20#define IDE_GD_VERSION "1.18"
21
806f80a6
BZ
22/* module parameters */
23static unsigned long debug_mask;
24module_param(debug_mask, ulong, 0644);
25
5fef0e5c
BZ
26static DEFINE_MUTEX(ide_disk_ref_mutex);
27
28static void ide_disk_release(struct kref *);
29
30static struct ide_disk_obj *ide_disk_get(struct gendisk *disk)
31{
32 struct ide_disk_obj *idkp = NULL;
33
34 mutex_lock(&ide_disk_ref_mutex);
35 idkp = ide_drv_g(disk, ide_disk_obj);
36 if (idkp) {
37 if (ide_device_get(idkp->drive))
38 idkp = NULL;
39 else
40 kref_get(&idkp->kref);
41 }
42 mutex_unlock(&ide_disk_ref_mutex);
43 return idkp;
44}
45
46static void ide_disk_put(struct ide_disk_obj *idkp)
47{
48 ide_drive_t *drive = idkp->drive;
49
50 mutex_lock(&ide_disk_ref_mutex);
51 kref_put(&idkp->kref, ide_disk_release);
52 ide_device_put(drive);
53 mutex_unlock(&ide_disk_ref_mutex);
54}
55
56sector_t ide_gd_capacity(ide_drive_t *drive)
57{
58 return drive->capacity64;
59}
60
61static int ide_gd_probe(ide_drive_t *);
62
63static void ide_gd_remove(ide_drive_t *drive)
64{
65 struct ide_disk_obj *idkp = drive->driver_data;
66 struct gendisk *g = idkp->disk;
67
68 ide_proc_unregister_driver(drive, idkp->driver);
69
70 del_gendisk(g);
71
806f80a6 72 drive->disk_ops->flush(drive);
5fef0e5c
BZ
73
74 ide_disk_put(idkp);
75}
76
77static void ide_disk_release(struct kref *kref)
78{
79 struct ide_disk_obj *idkp = to_ide_drv(kref, ide_disk_obj);
80 ide_drive_t *drive = idkp->drive;
81 struct gendisk *g = idkp->disk;
82
806f80a6 83 drive->disk_ops = NULL;
5fef0e5c
BZ
84 drive->driver_data = NULL;
85 g->private_data = NULL;
86 put_disk(g);
87 kfree(idkp);
88}
89
90/*
91 * On HPA drives the capacity needs to be
92 * reinitilized on resume otherwise the disk
93 * can not be used and a hard reset is required
94 */
95static void ide_gd_resume(ide_drive_t *drive)
96{
97 if (ata_id_hpa_enabled(drive->id))
806f80a6 98 (void)drive->disk_ops->get_capacity(drive);
5fef0e5c
BZ
99}
100
101static void ide_gd_shutdown(ide_drive_t *drive)
102{
103#ifdef CONFIG_ALPHA
104 /* On Alpha, halt(8) doesn't actually turn the machine off,
105 it puts you into the sort of firmware monitor. Typically,
106 it's used to boot another kernel image, so it's not much
107 different from reboot(8). Therefore, we don't need to
108 spin down the disk in this case, especially since Alpha
109 firmware doesn't handle disks in standby mode properly.
110 On the other hand, it's reasonably safe to turn the power
111 off when the shutdown process reaches the firmware prompt,
112 as the firmware initialization takes rather long time -
113 at least 10 seconds, which should be sufficient for
114 the disk to expire its write cache. */
115 if (system_state != SYSTEM_POWER_OFF) {
116#else
117 if (system_state == SYSTEM_RESTART) {
118#endif
806f80a6 119 drive->disk_ops->flush(drive);
5fef0e5c
BZ
120 return;
121 }
122
123 printk(KERN_INFO "Shutdown: %s\n", drive->name);
124
125 drive->gendev.bus->suspend(&drive->gendev, PMSG_SUSPEND);
126}
127
79cb3803
BZ
128#ifdef CONFIG_IDE_PROC_FS
129static ide_proc_entry_t *ide_disk_proc_entries(ide_drive_t *drive)
130{
806f80a6 131 return (drive->media == ide_disk) ? ide_disk_proc : ide_floppy_proc;
79cb3803
BZ
132}
133
134static const struct ide_proc_devset *ide_disk_proc_devsets(ide_drive_t *drive)
135{
806f80a6
BZ
136 return (drive->media == ide_disk) ? ide_disk_settings
137 : ide_floppy_settings;
79cb3803
BZ
138}
139#endif
140
806f80a6
BZ
141static ide_startstop_t ide_gd_do_request(ide_drive_t *drive,
142 struct request *rq, sector_t sector)
143{
144 return drive->disk_ops->do_request(drive, rq, sector);
145}
146
147static int ide_gd_end_request(ide_drive_t *drive, int uptodate, int nrsecs)
148{
149 return drive->disk_ops->end_request(drive, uptodate, nrsecs);
150}
151
7f3c868b 152static struct ide_driver ide_gd_driver = {
5fef0e5c
BZ
153 .gen_driver = {
154 .owner = THIS_MODULE,
806f80a6 155 .name = "ide-gd",
5fef0e5c
BZ
156 .bus = &ide_bus_type,
157 },
158 .probe = ide_gd_probe,
159 .remove = ide_gd_remove,
160 .resume = ide_gd_resume,
161 .shutdown = ide_gd_shutdown,
162 .version = IDE_GD_VERSION,
806f80a6
BZ
163 .do_request = ide_gd_do_request,
164 .end_request = ide_gd_end_request,
5fef0e5c 165#ifdef CONFIG_IDE_PROC_FS
79cb3803
BZ
166 .proc_entries = ide_disk_proc_entries,
167 .proc_devsets = ide_disk_proc_devsets,
5fef0e5c
BZ
168#endif
169};
170
b2f21e05 171static int ide_gd_open(struct block_device *bdev, fmode_t mode)
5fef0e5c 172{
b2f21e05 173 struct gendisk *disk = bdev->bd_disk;
5fef0e5c
BZ
174 struct ide_disk_obj *idkp;
175 ide_drive_t *drive;
806f80a6 176 int ret = 0;
5fef0e5c
BZ
177
178 idkp = ide_disk_get(disk);
179 if (idkp == NULL)
180 return -ENXIO;
181
182 drive = idkp->drive;
183
806f80a6
BZ
184 ide_debug_log(IDE_DBG_FUNC, "Call %s\n", __func__);
185
5fef0e5c
BZ
186 idkp->openers++;
187
188 if ((drive->dev_flags & IDE_DFLAG_REMOVABLE) && idkp->openers == 1) {
806f80a6
BZ
189 drive->dev_flags &= ~IDE_DFLAG_FORMAT_IN_PROGRESS;
190 /* Just in case */
191
192 ret = drive->disk_ops->init_media(drive, disk);
193
194 /*
195 * Allow O_NDELAY to open a drive without a disk, or with an
196 * unreadable disk, so that we can get the format capacity
197 * of the drive or begin the format - Sam
198 */
b2f21e05 199 if (ret && (mode & FMODE_NDELAY) == 0) {
806f80a6
BZ
200 ret = -EIO;
201 goto out_put_idkp;
202 }
203
b2f21e05 204 if ((drive->dev_flags & IDE_DFLAG_WP) && (mode & FMODE_WRITE)) {
806f80a6
BZ
205 ret = -EROFS;
206 goto out_put_idkp;
207 }
208
5fef0e5c
BZ
209 /*
210 * Ignore the return code from door_lock,
211 * since the open() has already succeeded,
212 * and the door_lock is irrelevant at this point.
213 */
806f80a6 214 drive->disk_ops->set_doorlock(drive, disk, 1);
cedd120c 215 drive->dev_flags |= IDE_DFLAG_MEDIA_CHANGED;
b2f21e05 216 check_disk_change(bdev);
806f80a6
BZ
217 } else if (drive->dev_flags & IDE_DFLAG_FORMAT_IN_PROGRESS) {
218 ret = -EBUSY;
219 goto out_put_idkp;
5fef0e5c
BZ
220 }
221 return 0;
806f80a6
BZ
222
223out_put_idkp:
224 idkp->openers--;
225 ide_disk_put(idkp);
226 return ret;
5fef0e5c
BZ
227}
228
b2f21e05 229static int ide_gd_release(struct gendisk *disk, fmode_t mode)
5fef0e5c 230{
5fef0e5c
BZ
231 struct ide_disk_obj *idkp = ide_drv_g(disk, ide_disk_obj);
232 ide_drive_t *drive = idkp->drive;
233
806f80a6
BZ
234 ide_debug_log(IDE_DBG_FUNC, "Call %s\n", __func__);
235
5fef0e5c 236 if (idkp->openers == 1)
806f80a6 237 drive->disk_ops->flush(drive);
5fef0e5c 238
806f80a6
BZ
239 if ((drive->dev_flags & IDE_DFLAG_REMOVABLE) && idkp->openers == 1) {
240 drive->disk_ops->set_doorlock(drive, disk, 0);
241 drive->dev_flags &= ~IDE_DFLAG_FORMAT_IN_PROGRESS;
242 }
5fef0e5c
BZ
243
244 idkp->openers--;
245
246 ide_disk_put(idkp);
247
248 return 0;
249}
250
251static int ide_gd_getgeo(struct block_device *bdev, struct hd_geometry *geo)
252{
253 struct ide_disk_obj *idkp = ide_drv_g(bdev->bd_disk, ide_disk_obj);
254 ide_drive_t *drive = idkp->drive;
255
256 geo->heads = drive->bios_head;
257 geo->sectors = drive->bios_sect;
258 geo->cylinders = (u16)drive->bios_cyl; /* truncate */
259 return 0;
260}
261
262static int ide_gd_media_changed(struct gendisk *disk)
263{
264 struct ide_disk_obj *idkp = ide_drv_g(disk, ide_disk_obj);
265 ide_drive_t *drive = idkp->drive;
cedd120c 266 int ret;
5fef0e5c
BZ
267
268 /* do not scan partitions twice if this is a removable device */
269 if (drive->dev_flags & IDE_DFLAG_ATTACH) {
270 drive->dev_flags &= ~IDE_DFLAG_ATTACH;
271 return 0;
272 }
273
cedd120c
BZ
274 ret = !!(drive->dev_flags & IDE_DFLAG_MEDIA_CHANGED);
275 drive->dev_flags &= ~IDE_DFLAG_MEDIA_CHANGED;
276
277 return ret;
5fef0e5c
BZ
278}
279
280static int ide_gd_revalidate_disk(struct gendisk *disk)
281{
282 struct ide_disk_obj *idkp = ide_drv_g(disk, ide_disk_obj);
52ebb438
BP
283 ide_drive_t *drive = idkp->drive;
284
285 if (ide_gd_media_changed(disk))
286 drive->disk_ops->get_capacity(drive);
287
288 set_capacity(disk, ide_gd_capacity(drive));
5fef0e5c
BZ
289 return 0;
290}
291
b2f21e05 292static int ide_gd_ioctl(struct block_device *bdev, fmode_t mode,
806f80a6
BZ
293 unsigned int cmd, unsigned long arg)
294{
806f80a6
BZ
295 struct ide_disk_obj *idkp = ide_drv_g(bdev->bd_disk, ide_disk_obj);
296 ide_drive_t *drive = idkp->drive;
297
b2f21e05 298 return drive->disk_ops->ioctl(drive, bdev, mode, cmd, arg);
806f80a6
BZ
299}
300
5fef0e5c
BZ
301static struct block_device_operations ide_gd_ops = {
302 .owner = THIS_MODULE,
b2f21e05
AV
303 .open = ide_gd_open,
304 .release = ide_gd_release,
305 .locked_ioctl = ide_gd_ioctl,
5fef0e5c
BZ
306 .getgeo = ide_gd_getgeo,
307 .media_changed = ide_gd_media_changed,
308 .revalidate_disk = ide_gd_revalidate_disk
309};
310
311static int ide_gd_probe(ide_drive_t *drive)
312{
806f80a6 313 const struct ide_disk_ops *disk_ops = NULL;
5fef0e5c
BZ
314 struct ide_disk_obj *idkp;
315 struct gendisk *g;
316
317 /* strstr("foo", "") is non-NULL */
806f80a6
BZ
318 if (!strstr("ide-gd", drive->driver_req))
319 goto failed;
320
321#ifdef CONFIG_IDE_GD_ATA
322 if (drive->media == ide_disk)
323 disk_ops = &ide_ata_disk_ops;
324#endif
325#ifdef CONFIG_IDE_GD_ATAPI
326 if (drive->media == ide_floppy)
327 disk_ops = &ide_atapi_disk_ops;
328#endif
329 if (disk_ops == NULL)
5fef0e5c
BZ
330 goto failed;
331
806f80a6
BZ
332 if (disk_ops->check(drive, DRV_NAME) == 0) {
333 printk(KERN_ERR PFX "%s: not supported by this driver\n",
334 drive->name);
5fef0e5c 335 goto failed;
806f80a6 336 }
5fef0e5c
BZ
337
338 idkp = kzalloc(sizeof(*idkp), GFP_KERNEL);
806f80a6
BZ
339 if (!idkp) {
340 printk(KERN_ERR PFX "%s: can't allocate a disk structure\n",
341 drive->name);
5fef0e5c 342 goto failed;
806f80a6 343 }
5fef0e5c
BZ
344
345 g = alloc_disk_node(IDE_DISK_MINORS, hwif_to_node(drive->hwif));
346 if (!g)
347 goto out_free_idkp;
348
349 ide_init_disk(g, drive);
350
351 kref_init(&idkp->kref);
352
353 idkp->drive = drive;
354 idkp->driver = &ide_gd_driver;
355 idkp->disk = g;
356
357 g->private_data = &idkp->driver;
358
359 drive->driver_data = idkp;
806f80a6
BZ
360 drive->debug_mask = debug_mask;
361 drive->disk_ops = disk_ops;
5fef0e5c 362
806f80a6 363 disk_ops->setup(drive);
5fef0e5c
BZ
364
365 set_capacity(g, ide_gd_capacity(drive));
366
367 g->minors = IDE_DISK_MINORS;
368 g->driverfs_dev = &drive->gendev;
369 g->flags |= GENHD_FL_EXT_DEVT;
370 if (drive->dev_flags & IDE_DFLAG_REMOVABLE)
371 g->flags = GENHD_FL_REMOVABLE;
372 g->fops = &ide_gd_ops;
373 add_disk(g);
374 return 0;
375
376out_free_idkp:
377 kfree(idkp);
378failed:
379 return -ENODEV;
380}
381
382static int __init ide_gd_init(void)
383{
806f80a6 384 printk(KERN_INFO DRV_NAME " driver " IDE_GD_VERSION "\n");
5fef0e5c
BZ
385 return driver_register(&ide_gd_driver.gen_driver);
386}
387
388static void __exit ide_gd_exit(void)
389{
390 driver_unregister(&ide_gd_driver.gen_driver);
391}
392
393MODULE_ALIAS("ide:*m-disk*");
394MODULE_ALIAS("ide-disk");
806f80a6
BZ
395MODULE_ALIAS("ide:*m-floppy*");
396MODULE_ALIAS("ide-floppy");
5fef0e5c
BZ
397module_init(ide_gd_init);
398module_exit(ide_gd_exit);
399MODULE_LICENSE("GPL");
806f80a6 400MODULE_DESCRIPTION("generic ATA/ATAPI disk driver");
This page took 0.145311 seconds and 5 git commands to generate.