2 Copyright (C) 2013 Carl Hetherington <cth@carlh.net>
4 This program is free software; you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation; either version 2 of the License, or
7 (at your option) any later version.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program; if not, write to the Free Software
16 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
22 #include "file_group.h"
23 #include "exceptions.h"
29 FileGroup::FileGroup ()
36 FileGroup::FileGroup (boost::filesystem::path p)
44 FileGroup::FileGroup (vector<boost::filesystem::path> const & p)
53 FileGroup::~FileGroup ()
56 fclose (_current_file);
61 FileGroup::set_paths (vector<boost::filesystem::path> const & p)
68 /** Ensure that the given path index in the content is the _current_file */
70 FileGroup::ensure_open_path (size_t p) const
72 if (_current_file && _current_path == p) {
78 fclose (_current_file);
82 _current_file = fopen_boost (_paths[_current_path], "rb");
83 if (_current_file == 0) {
84 throw OpenFileError (_paths[_current_path]);
89 FileGroup::seek (int64_t pos, int whence) const
91 /* Convert pos to `full_pos', which is an offset from the start
100 for (size_t i = 0; i < _current_path; ++i) {
101 full_pos += boost::filesystem::file_size (_paths[i]);
103 #ifdef DCPOMATIC_WINDOWS
104 full_pos += _ftelli64 (_current_file);
106 full_pos += ftell (_current_file);
111 full_pos = length() - pos;
115 /* Seek to full_pos */
117 int64_t sub_pos = full_pos;
118 while (i < _paths.size ()) {
119 boost::uintmax_t len = boost::filesystem::file_size (_paths[i]);
120 if (sub_pos < int64_t (len)) {
127 if (i == _paths.size ()) {
131 ensure_open_path (i);
132 dcpomatic_fseek (_current_file, sub_pos, SEEK_SET);
136 /** Try to read some data from the current position into a buffer.
137 * @param buffer Buffer to write data into.
138 * @param amount Number of bytes to read.
139 * @return Number of bytes read, or -1 in the case of error.
142 FileGroup::read (uint8_t* buffer, int amount) const
146 int const this_time = fread (buffer + read, 1, amount - read, _current_file);
148 if (read == amount) {
153 /* See if there is another file to use */
154 if ((_current_path + 1) >= _paths.size()) {
157 ensure_open_path (_current_path + 1);
164 FileGroup::length () const
167 for (size_t i = 0; i < _paths.size(); ++i) {
168 len += boost::filesystem::file_size (_paths[i]);