blob: abf7bdf46e05f8f40d76f87f5bd371551b39afea [file] [log] [blame]
Dave Chapman4b7e1e02006-12-13 09:02:18 +00001/***************************************************************************
2 * __________ __ ___.
3 * Open \______ \ ____ ____ | | _\_ |__ _______ ___
4 * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
5 * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
6 * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
7 * \/ \/ \/ \/ \/
8 * $Id$
9 *
Dave Chapmane332f4c2007-02-05 01:20:20 +000010 * Copyright (C) 2006-2007 Dave Chapman
Dave Chapman4b7e1e02006-12-13 09:02:18 +000011 *
12 * error(), lock_volume() and unlock_volume() functions and inspiration taken
13 * from:
14 * RawDisk - Direct Disk Read/Write Access for NT/2000/XP
15 * Copyright (c) 2003 Jan Kiszka
16 * http://www.stud.uni-hannover.de/user/73174/RawDisk/
17 *
Daniel Stenberg2acc0ac2008-06-28 18:10:04 +000018 * This program is free software; you can redistribute it and/or
19 * modify it under the terms of the GNU General Public License
20 * as published by the Free Software Foundation; either version 2
21 * of the License, or (at your option) any later version.
Dave Chapman4b7e1e02006-12-13 09:02:18 +000022 *
23 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
24 * KIND, either express or implied.
25 *
26 ****************************************************************************/
27
28#include <stdio.h>
29#include <unistd.h>
30#include <fcntl.h>
31#include <string.h>
32#include <stdlib.h>
33#include <sys/types.h>
34#include <sys/stat.h>
Dave Chapman4b7e1e02006-12-13 09:02:18 +000035#include <windows.h>
Dave Chapman1eca02d2009-08-04 20:32:30 +000036#include <stddef.h>
Dave Chapman4b7e1e02006-12-13 09:02:18 +000037#include <winioctl.h>
Dave Chapman4b7e1e02006-12-13 09:02:18 +000038
39#include "ipodio.h"
40
41static int lock_volume(HANDLE hDisk)
42{
43 DWORD dummy;
44
45 return DeviceIoControl(hDisk, FSCTL_LOCK_VOLUME, NULL, 0, NULL, 0,
46 &dummy, NULL);
47}
48
49static int unlock_volume(HANDLE hDisk)
50{
51 DWORD dummy;
52
53 return DeviceIoControl(hDisk, FSCTL_UNLOCK_VOLUME, NULL, 0, NULL, 0,
54 &dummy, NULL);
55}
56
Dominik Riebeling3e489c12009-11-08 13:38:10 +000057void ipod_print_error(char* msg)
Dave Chapman4b7e1e02006-12-13 09:02:18 +000058{
Dominik Riebeling8b32a2d2009-09-20 17:03:58 +000059 LPSTR pMsgBuf = NULL;
Dave Chapman4b7e1e02006-12-13 09:02:18 +000060
61 printf(msg);
Dominik Riebeling8b32a2d2009-09-20 17:03:58 +000062 FormatMessageA(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM |
Dave Chapman4b7e1e02006-12-13 09:02:18 +000063 FORMAT_MESSAGE_IGNORE_INSERTS, NULL, GetLastError(),
Dominik Riebeling8b32a2d2009-09-20 17:03:58 +000064 MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), pMsgBuf,
Dave Chapman4b7e1e02006-12-13 09:02:18 +000065 0, NULL);
66 printf(pMsgBuf);
67 LocalFree(pMsgBuf);
68}
69
Dave Chapman31aa4522007-02-04 11:42:11 +000070int ipod_open(struct ipod_t* ipod, int silent)
Dave Chapman4b7e1e02006-12-13 09:02:18 +000071{
72 DISK_GEOMETRY_EX diskgeometry_ex;
73 DISK_GEOMETRY diskgeometry;
74 unsigned long n;
75
Magnus Holmgren8a7c0ee2007-11-06 19:28:14 +000076 ipod->dh = CreateFileA(ipod->diskname, GENERIC_READ,
Dave Chapman4b7e1e02006-12-13 09:02:18 +000077 FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING,
78 FILE_FLAG_WRITE_THROUGH | FILE_FLAG_NO_BUFFERING, NULL);
79
Dave Chapman31aa4522007-02-04 11:42:11 +000080 if (ipod->dh == INVALID_HANDLE_VALUE) {
Dominik Riebeling3e489c12009-11-08 13:38:10 +000081 if (!silent) ipod_print_error(" Error opening disk: ");
Dominik Riebeling194b2ca2008-05-04 11:59:04 +000082 if(GetLastError() == ERROR_ACCESS_DENIED)
83 return -2;
84 else
85 return -1;
Dave Chapman4b7e1e02006-12-13 09:02:18 +000086 }
87
Dave Chapman31aa4522007-02-04 11:42:11 +000088 if (!lock_volume(ipod->dh)) {
Dominik Riebeling3e489c12009-11-08 13:38:10 +000089 if (!silent) ipod_print_error(" Error locking disk: ");
Dave Chapman4b7e1e02006-12-13 09:02:18 +000090 return -1;
91 }
92
Dave Chapman56780e32007-06-16 22:32:57 +000093 /* Defaults */
94 ipod->num_heads = 0;
95 ipod->sectors_per_track = 0;
96
Dave Chapman31aa4522007-02-04 11:42:11 +000097 if (!DeviceIoControl(ipod->dh,
Dave Chapman4b7e1e02006-12-13 09:02:18 +000098 IOCTL_DISK_GET_DRIVE_GEOMETRY_EX,
99 NULL,
100 0,
101 &diskgeometry_ex,
102 sizeof(diskgeometry_ex),
103 &n,
104 NULL)) {
Dave Chapman31aa4522007-02-04 11:42:11 +0000105 if (!DeviceIoControl(ipod->dh,
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000106 IOCTL_DISK_GET_DRIVE_GEOMETRY,
107 NULL,
108 0,
109 &diskgeometry,
110 sizeof(diskgeometry),
111 &n,
112 NULL)) {
Dominik Riebeling3e489c12009-11-08 13:38:10 +0000113 if (!silent) ipod_print_error(" Error reading disk geometry: ");
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000114 return -1;
115 } else {
Dave Chapman56780e32007-06-16 22:32:57 +0000116 ipod->sector_size = diskgeometry.BytesPerSector;
117 ipod->num_heads = diskgeometry.TracksPerCylinder;
118 ipod->sectors_per_track = diskgeometry.SectorsPerTrack;
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000119 }
120 } else {
Dave Chapman56780e32007-06-16 22:32:57 +0000121 ipod->sector_size = diskgeometry_ex.Geometry.BytesPerSector;
122 ipod->num_heads = diskgeometry_ex.Geometry.TracksPerCylinder;
123 ipod->sectors_per_track = diskgeometry_ex.Geometry.SectorsPerTrack;
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000124 }
125
126 return 0;
127}
128
Dave Chapman31aa4522007-02-04 11:42:11 +0000129int ipod_reopen_rw(struct ipod_t* ipod)
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000130{
131 /* Close existing file and re-open for writing */
Dave Chapman31aa4522007-02-04 11:42:11 +0000132 unlock_volume(ipod->dh);
133 CloseHandle(ipod->dh);
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000134
Magnus Holmgren8a7c0ee2007-11-06 19:28:14 +0000135 ipod->dh = CreateFileA(ipod->diskname, GENERIC_READ | GENERIC_WRITE,
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000136 FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING,
137 FILE_FLAG_WRITE_THROUGH | FILE_FLAG_NO_BUFFERING, NULL);
138
Dave Chapman31aa4522007-02-04 11:42:11 +0000139 if (ipod->dh == INVALID_HANDLE_VALUE) {
Dominik Riebeling3e489c12009-11-08 13:38:10 +0000140 ipod_print_error(" Error opening disk: ");
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000141 return -1;
142 }
143
Dave Chapman31aa4522007-02-04 11:42:11 +0000144 if (!lock_volume(ipod->dh)) {
Dominik Riebeling3e489c12009-11-08 13:38:10 +0000145 ipod_print_error(" Error locking disk: ");
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000146 return -1;
147 }
148
149 return 0;
150}
151
Dave Chapman31aa4522007-02-04 11:42:11 +0000152int ipod_close(struct ipod_t* ipod)
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000153{
Dave Chapman31aa4522007-02-04 11:42:11 +0000154 unlock_volume(ipod->dh);
155 CloseHandle(ipod->dh);
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000156 return 0;
157}
158
159int ipod_alloc_buffer(unsigned char** sectorbuf, int bufsize)
160{
161 /* The ReadFile function requires a memory buffer aligned to a multiple of
162 the disk sector size. */
Dave Chapmane8483752006-12-14 22:17:38 +0000163 *sectorbuf = (unsigned char*)VirtualAlloc(NULL, bufsize, MEM_COMMIT, PAGE_READWRITE);
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000164 if (*sectorbuf == NULL) {
Dominik Riebeling3e489c12009-11-08 13:38:10 +0000165 ipod_print_error(" Error allocating a buffer: ");
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000166 return -1;
167 }
168 return 0;
169}
170
Dave Chapman31aa4522007-02-04 11:42:11 +0000171int ipod_seek(struct ipod_t* ipod, unsigned long pos)
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000172{
Dave Chapman31aa4522007-02-04 11:42:11 +0000173 if (SetFilePointer(ipod->dh, pos, NULL, FILE_BEGIN)==0xffffffff) {
Dominik Riebeling3e489c12009-11-08 13:38:10 +0000174 ipod_print_error(" Seek error ");
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000175 return -1;
176 }
177 return 0;
178}
179
Dave Chapman2cc80f52007-07-29 21:19:14 +0000180ssize_t ipod_read(struct ipod_t* ipod, unsigned char* buf, int nbytes)
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000181{
182 unsigned long count;
183
Dave Chapman31aa4522007-02-04 11:42:11 +0000184 if (!ReadFile(ipod->dh, buf, nbytes, &count, NULL)) {
Dominik Riebeling3e489c12009-11-08 13:38:10 +0000185 ipod_print_error(" Error reading from disk: ");
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000186 return -1;
187 }
188
189 return count;
190}
191
Dave Chapman2cc80f52007-07-29 21:19:14 +0000192ssize_t ipod_write(struct ipod_t* ipod, unsigned char* buf, int nbytes)
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000193{
194 unsigned long count;
195
Dave Chapman31aa4522007-02-04 11:42:11 +0000196 if (!WriteFile(ipod->dh, buf, nbytes, &count, NULL)) {
Dominik Riebeling3e489c12009-11-08 13:38:10 +0000197 ipod_print_error(" Error writing to disk: ");
Dave Chapman4b7e1e02006-12-13 09:02:18 +0000198 return -1;
199 }
200
201 return count;
202}
Dave Chapman1eca02d2009-08-04 20:32:30 +0000203