blob: 1b88fd5df05d6d7f6694a3d9284162d4ed386a0b [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * lowlevel.c
3 *
4 * PURPOSE
5 * Low Level Device Routines for the UDF filesystem
6 *
Linus Torvalds1da177e2005-04-16 15:20:36 -07007 * COPYRIGHT
8 * This file is distributed under the terms of the GNU General Public
9 * License (GPL). Copies of the GPL can be obtained from:
10 * ftp://prep.ai.mit.edu/pub/gnu/GPL
11 * Each contributing author retains all rights to their own work.
12 *
13 * (C) 1999-2001 Ben Fennema
14 *
15 * HISTORY
16 *
17 * 03/26/99 blf Created.
18 */
19
20#include "udfdecl.h"
21
22#include <linux/blkdev.h>
23#include <linux/cdrom.h>
24#include <asm/uaccess.h>
25
Linus Torvalds1da177e2005-04-16 15:20:36 -070026#include "udf_sb.h"
27
Cyrill Gorcunovcb00ea32007-07-19 01:47:43 -070028unsigned int udf_get_last_session(struct super_block *sb)
Linus Torvalds1da177e2005-04-16 15:20:36 -070029{
30 struct cdrom_multisession ms_info;
31 unsigned int vol_desc_start;
32 struct block_device *bdev = sb->s_bdev;
33 int i;
34
Cyrill Gorcunovcb00ea32007-07-19 01:47:43 -070035 vol_desc_start = 0;
36 ms_info.addr_format = CDROM_LBA;
37 i = ioctl_by_bdev(bdev, CDROMMULTISESSION, (unsigned long)&ms_info);
Linus Torvalds1da177e2005-04-16 15:20:36 -070038
39#define WE_OBEY_THE_WRITTEN_STANDARDS 1
40
Cyrill Gorcunovcb00ea32007-07-19 01:47:43 -070041 if (i == 0) {
Linus Torvalds1da177e2005-04-16 15:20:36 -070042 udf_debug("XA disk: %s, vol_desc_start=%d\n",
Cyrill Gorcunovcb00ea32007-07-19 01:47:43 -070043 (ms_info.xa_flag ? "yes" : "no"), ms_info.addr.lba);
Linus Torvalds1da177e2005-04-16 15:20:36 -070044#if WE_OBEY_THE_WRITTEN_STANDARDS
Cyrill Gorcunov28de7942007-07-21 04:37:18 -070045 if (ms_info.xa_flag) /* necessary for a valid ms_info.addr */
Linus Torvalds1da177e2005-04-16 15:20:36 -070046#endif
47 vol_desc_start = ms_info.addr.lba;
Cyrill Gorcunovcb00ea32007-07-19 01:47:43 -070048 } else {
Linus Torvalds1da177e2005-04-16 15:20:36 -070049 udf_debug("CDROMMULTISESSION not supported: rc=%d\n", i);
50 }
51 return vol_desc_start;
52}
53
Cyrill Gorcunovcb00ea32007-07-19 01:47:43 -070054unsigned long udf_get_last_block(struct super_block *sb)
Linus Torvalds1da177e2005-04-16 15:20:36 -070055{
56 struct block_device *bdev = sb->s_bdev;
57 unsigned long lblock = 0;
58
Jan Kara24a5d592009-06-18 12:33:16 +020059 /*
60 * ioctl failed or returned obviously bogus value?
61 * Try using the device size...
62 */
63 if (ioctl_by_bdev(bdev, CDROM_LAST_WRITTEN, (unsigned long) &lblock) ||
64 lblock == 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -070065 lblock = bdev->bd_inode->i_size >> sb->s_blocksize_bits;
66
67 if (lblock)
68 return lblock - 1;
69 else
70 return 0;
71}