/* Copyright (C) 2012-2020 Carl Hetherington This file is part of DCP-o-matic. DCP-o-matic is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. DCP-o-matic is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with DCP-o-matic. If not, see . */ /** @file src/lib/util.cc * @brief Some utility functions and classes. */ #include "util.h" #include "exceptions.h" #include "dcp_content_type.h" #include "filter.h" #include "cinema_sound_processor.h" #include "config.h" #include "ratio.h" #include "job.h" #include "cross.h" #include "video_content.h" #include "rect.h" #include "digester.h" #include "audio_processor.h" #include "crypto.h" #include "compose.hpp" #include "audio_buffers.h" #include "string_text.h" #include "font.h" #include "render_text.h" #include "ffmpeg_image_proxy.h" #include "image.h" #include "text_decoder.h" #include "job_manager.h" #include #include #include #include #include #include extern "C" { #include #include #include } #include #include #include #include #include #include #include #include #ifdef DCPOMATIC_WINDOWS #include #include #endif #include #include #include #include #include #include #ifdef DCPOMATIC_POSIX #include #include #endif #include "i18n.h" using std::string; using std::wstring; using std::setfill; using std::ostream; using std::endl; using std::vector; using std::min; using std::max; using std::map; using std::list; using std::multimap; using std::istream; using std::pair; using std::cout; using std::bad_alloc; using std::set_terminate; using std::make_pair; using boost::shared_ptr; using boost::thread; using boost::optional; using boost::lexical_cast; using boost::bad_lexical_cast; using boost::scoped_array; using dcp::Size; using dcp::raw_convert; using dcp::locale_convert; using namespace dcpomatic; /** Path to our executable, required by the stacktrace stuff and filled * in during App::onInit(). */ string program_name; bool is_batch_converter = false; static boost::thread::id ui_thread; static boost::filesystem::path backtrace_file; /** Convert some number of seconds to a string representation * in hours, minutes and seconds. * * @param s Seconds. * @return String of the form H:M:S (where H is hours, M * is minutes and S is seconds). */ string seconds_to_hms (int s) { int m = s / 60; s -= (m * 60); int h = m / 60; m -= (h * 60); char buffer[64]; snprintf (buffer, sizeof(buffer), "%d:%02d:%02d", h, m, s); return buffer; } string time_to_hmsf (DCPTime time, Frame rate) { Frame f = time.frames_round (rate); int s = f / rate; f -= (s * rate); int m = s / 60; s -= m * 60; int h = m / 60; m -= h * 60; char buffer[64]; snprintf (buffer, sizeof(buffer), "%d:%02d:%02d.%d", h, m, s, static_cast(f)); return buffer; } /** @param s Number of seconds. * @return String containing an approximate description of s (e.g. "about 2 hours") */ string seconds_to_approximate_hms (int s) { int m = s / 60; s -= (m * 60); int h = m / 60; m -= (h * 60); string ap; bool hours = h > 0; bool minutes = h < 6 && m > 0; bool seconds = h == 0 && m < 10 && s > 0; if (m > 30 && !minutes) { /* round up the hours */ ++h; } if (s > 30 && !seconds) { /* round up the minutes */ ++m; if (m == 60) { m = 0; minutes = false; ++h; } } if (hours) { /// TRANSLATORS: h here is an abbreviation for hours ap += locale_convert(h) + _("h"); if (minutes || seconds) { ap += N_(" "); } } if (minutes) { /// TRANSLATORS: m here is an abbreviation for minutes ap += locale_convert(m) + _("m"); if (seconds) { ap += N_(" "); } } if (seconds) { /* Seconds */ /// TRANSLATORS: s here is an abbreviation for seconds ap += locale_convert(s) + _("s"); } return ap; } double seconds (struct timeval t) { return t.tv_sec + (double (t.tv_usec) / 1e6); } #ifdef DCPOMATIC_WINDOWS /** Resolve symbol name and source location given the path to the executable */ int addr2line (void const * const addr) { char addr2line_cmd[512] = { 0 }; sprintf (addr2line_cmd, "addr2line -f -p -e %.256s %p > %s", program_name.c_str(), addr, backtrace_file.string().c_str()); return system(addr2line_cmd); } /** This is called when C signals occur on Windows (e.g. SIGSEGV) * (NOT C++ exceptions!). We write a backtrace to backtrace_file by dark means. * Adapted from code here: http://spin.atomicobject.com/2013/01/13/exceptions-stack-traces-c/ */ LONG WINAPI exception_handler(struct _EXCEPTION_POINTERS * info) { FILE* f = fopen_boost (backtrace_file, "w"); fprintf (f, "C-style exception %d\n", info->ExceptionRecord->ExceptionCode); fclose(f); if (info->ExceptionRecord->ExceptionCode != EXCEPTION_STACK_OVERFLOW) { CONTEXT* context = info->ContextRecord; SymInitialize (GetCurrentProcess (), 0, true); STACKFRAME frame = { 0 }; /* setup initial stack frame */ #if _WIN64 frame.AddrPC.Offset = context->Rip; frame.AddrStack.Offset = context->Rsp; frame.AddrFrame.Offset = context->Rbp; #else frame.AddrPC.Offset = context->Eip; frame.AddrStack.Offset = context->Esp; frame.AddrFrame.Offset = context->Ebp; #endif frame.AddrPC.Mode = AddrModeFlat; frame.AddrStack.Mode = AddrModeFlat; frame.AddrFrame.Mode = AddrModeFlat; while ( StackWalk ( IMAGE_FILE_MACHINE_I386, GetCurrentProcess (), GetCurrentThread (), &frame, context, 0, SymFunctionTableAccess, SymGetModuleBase, 0 ) ) { addr2line((void *) frame.AddrPC.Offset); } } else { #ifdef _WIN64 addr2line ((void *) info->ContextRecord->Rip); #else addr2line ((void *) info->ContextRecord->Eip); #endif } return EXCEPTION_CONTINUE_SEARCH; } #endif void set_backtrace_file (boost::filesystem::path p) { backtrace_file = p; } /** This is called when there is an unhandled exception. Any * backtrace in this function is useless on Windows as the stack has * already been unwound from the throw; we have the gdb wrap hack to * cope with that. */ void terminate () { try { static bool tried_throw = false; // try once to re-throw currently active exception if (!tried_throw) { tried_throw = true; throw; } } catch (const std::exception &e) { std::cerr << __FUNCTION__ << " caught unhandled exception. what(): " << e.what() << std::endl; } catch (...) { std::cerr << __FUNCTION__ << " caught unknown/unhandled exception." << std::endl; } abort(); } void dcpomatic_setup_path_encoding () { #ifdef DCPOMATIC_WINDOWS /* Dark voodoo which, I think, gets boost::filesystem::path to correctly convert UTF-8 strings to paths, and also paths back to UTF-8 strings (on path::string()). After this, constructing boost::filesystem::paths from strings converts from UTF-8 to UTF-16 inside the path. Then path::string().c_str() gives UTF-8 and path::c_str() gives UTF-16. This is all Windows-only. AFAICT Linux/OS X use UTF-8 everywhere, so things are much simpler. */ std::locale::global (boost::locale::generator().generate ("")); boost::filesystem::path::imbue (std::locale ()); #endif } /** Call the required functions to set up DCP-o-matic's static arrays, etc. * Must be called from the UI thread, if there is one. */ void dcpomatic_setup () { #ifdef DCPOMATIC_WINDOWS boost::filesystem::path p = g_get_user_config_dir (); p /= "backtrace.txt"; set_backtrace_file (p); SetUnhandledExceptionFilter(exception_handler); #endif av_register_all (); avfilter_register_all (); #ifdef DCPOMATIC_OSX /* Add our library directory to the libltdl search path so that xmlsec can find xmlsec1-openssl. */ boost::filesystem::path lib = app_contents (); lib /= "Frameworks"; setenv ("LTDL_LIBRARY_PATH", lib.c_str (), 1); #endif set_terminate (terminate); #ifdef DCPOMATIC_WINDOWS putenv ("PANGOCAIRO_BACKEND=fontconfig"); putenv (String::compose("FONTCONFIG_PATH=%1", shared_path().string()).c_str()); #endif #ifdef DCPOMATIC_OSX setenv ("PANGOCAIRO_BACKEND", "fontconfig", 1); setenv ("FONTCONFIG_PATH", shared_path().string().c_str(), 1); #endif Pango::init (); dcp::init (); #if defined(DCPOMATIC_WINDOWS) || defined(DCPOMATIC_OSX) /* Render something to fontconfig to create its cache */ list subs; dcp::SubtitleString ss( optional(), false, false, false, dcp::Colour(), 42, 1, dcp::Time(), dcp::Time(), 0, dcp::HALIGN_CENTER, 0, dcp::VALIGN_CENTER, dcp::DIRECTION_LTR, "Hello dolly", dcp::NONE, dcp::Colour(), dcp::Time(), dcp::Time() ); subs.push_back (StringText(ss, 0)); render_text (subs, list >(), dcp::Size(640, 480), DCPTime(), 24); #endif Ratio::setup_ratios (); PresetColourConversion::setup_colour_conversion_presets (); VideoContentScale::setup_scales (); DCPContentType::setup_dcp_content_types (); Filter::setup_filters (); CinemaSoundProcessor::setup_cinema_sound_processors (); AudioProcessor::setup_audio_processors (); curl_global_init (CURL_GLOBAL_ALL); ui_thread = boost::this_thread::get_id (); } #ifdef DCPOMATIC_WINDOWS boost::filesystem::path mo_path () { wchar_t buffer[512]; GetModuleFileName (0, buffer, 512 * sizeof(wchar_t)); boost::filesystem::path p (buffer); p = p.parent_path (); p = p.parent_path (); p /= "locale"; return p; } #endif #ifdef DCPOMATIC_OSX boost::filesystem::path mo_path () { return "DCP-o-matic 2.app/Contents/Resources"; } #endif void dcpomatic_setup_gettext_i18n (string lang) { #ifdef DCPOMATIC_LINUX lang += ".UTF8"; #endif if (!lang.empty ()) { /* Override our environment language. Note that the caller must not free the string passed into putenv(). */ string s = String::compose ("LANGUAGE=%1", lang); putenv (strdup (s.c_str ())); s = String::compose ("LANG=%1", lang); putenv (strdup (s.c_str ())); s = String::compose ("LC_ALL=%1", lang); putenv (strdup (s.c_str ())); } setlocale (LC_ALL, ""); textdomain ("libdcpomatic2"); #if defined(DCPOMATIC_WINDOWS) || defined(DCPOMATIC_OSX) bindtextdomain ("libdcpomatic2", mo_path().string().c_str()); bind_textdomain_codeset ("libdcpomatic2", "UTF8"); #endif #ifdef DCPOMATIC_LINUX bindtextdomain ("libdcpomatic2", LINUX_LOCALE_PREFIX); #endif } /** Compute a digest of the first and last `size' bytes of a set of files. */ string digest_head_tail (vector files, boost::uintmax_t size) { boost::scoped_array buffer (new char[size]); Digester digester; /* Head */ boost::uintmax_t to_do = size; char* p = buffer.get (); int i = 0; while (i < int64_t (files.size()) && to_do > 0) { FILE* f = fopen_boost (files[i], "rb"); if (!f) { throw OpenFileError (files[i].string(), errno, OpenFileError::READ); } boost::uintmax_t this_time = min (to_do, boost::filesystem::file_size (files[i])); checked_fread (p, this_time, f, files[i]); p += this_time; to_do -= this_time; fclose (f); ++i; } digester.add (buffer.get(), size - to_do); /* Tail */ to_do = size; p = buffer.get (); i = files.size() - 1; while (i >= 0 && to_do > 0) { FILE* f = fopen_boost (files[i], "rb"); if (!f) { throw OpenFileError (files[i].string(), errno, OpenFileError::READ); } boost::uintmax_t this_time = min (to_do, boost::filesystem::file_size (files[i])); dcpomatic_fseek (f, -this_time, SEEK_END); checked_fread (p, this_time, f, files[i]); p += this_time; to_do -= this_time; fclose (f); --i; } digester.add (buffer.get(), size - to_do); return digester.get (); } /** Round a number up to the nearest multiple of another number. * @param c Index. * @param stride Array of numbers to round, indexed by c. * @param t Multiple to round to. * @return Rounded number. */ int stride_round_up (int c, int const * stride, int t) { int const a = stride[c] + (t - 1); return a - (a % t); } /** Trip an assert if the caller is not in the UI thread */ void ensure_ui_thread () { DCPOMATIC_ASSERT (boost::this_thread::get_id() == ui_thread); } string audio_channel_name (int c) { DCPOMATIC_ASSERT (MAX_DCP_AUDIO_CHANNELS == 16); /// TRANSLATORS: these are the names of audio channels; Lfe (sub) is the low-frequency /// enhancement channel (sub-woofer). string const channels[] = { _("Left"), _("Right"), _("Centre"), _("Lfe (sub)"), _("Left surround"), _("Right surround"), _("Hearing impaired"), _("Visually impaired"), _("Left centre"), _("Right centre"), _("Left rear surround"), _("Right rear surround"), _("D-BOX primary"), _("D-BOX secondary"), _("Unused"), _("Unused") }; return channels[c]; } string short_audio_channel_name (int c) { DCPOMATIC_ASSERT (MAX_DCP_AUDIO_CHANNELS == 16); /// TRANSLATORS: these are short names of audio channels; Lfe is the low-frequency /// enhancement channel (sub-woofer). HI is the hearing-impaired audio track and /// VI is the visually-impaired audio track (audio describe). DBP is the D-BOX /// primary channel and DBS is the D-BOX secondary channel. string const channels[] = { _("L"), _("R"), _("C"), _("Lfe"), _("Ls"), _("Rs"), _("HI"), _("VI"), _("Lc"), _("Rc"), _("BsL"), _("BsR"), _("DBP"), _("DBS"), "", "" }; return channels[c]; } bool valid_image_file (boost::filesystem::path f) { if (boost::starts_with (f.leaf().string(), "._")) { return false; } string ext = f.extension().string(); transform (ext.begin(), ext.end(), ext.begin(), ::tolower); return ( ext == ".tif" || ext == ".tiff" || ext == ".jpg" || ext == ".jpeg" || ext == ".png" || ext == ".bmp" || ext == ".tga" || ext == ".dpx" || ext == ".j2c" || ext == ".j2k" || ext == ".jp2" || ext == ".exr" || ext == ".jpf" ); } bool valid_sound_file (boost::filesystem::path f) { if (boost::starts_with (f.leaf().string(), "._")) { return false; } string ext = f.extension().string(); transform (ext.begin(), ext.end(), ext.begin(), ::tolower); return (ext == ".wav" || ext == ".mp3" || ext == ".aif" || ext == ".aiff"); } bool valid_j2k_file (boost::filesystem::path f) { string ext = f.extension().string(); transform (ext.begin(), ext.end(), ext.begin(), ::tolower); return (ext == ".j2k" || ext == ".j2c" || ext == ".jp2"); } string tidy_for_filename (string f) { boost::replace_if (f, boost::is_any_of ("\\/:"), '_'); return f; } dcp::Size fit_ratio_within (float ratio, dcp::Size full_frame) { if (ratio < full_frame.ratio ()) { return dcp::Size (lrintf (full_frame.height * ratio), full_frame.height); } return dcp::Size (full_frame.width, lrintf (full_frame.width / ratio)); } void * wrapped_av_malloc (size_t s) { void* p = av_malloc (s); if (!p) { throw bad_alloc (); } return p; } map split_get_request (string url) { enum { AWAITING_QUESTION_MARK, KEY, VALUE } state = AWAITING_QUESTION_MARK; map r; string k; string v; for (size_t i = 0; i < url.length(); ++i) { switch (state) { case AWAITING_QUESTION_MARK: if (url[i] == '?') { state = KEY; } break; case KEY: if (url[i] == '=') { v.clear (); state = VALUE; } else { k += url[i]; } break; case VALUE: if (url[i] == '&') { r.insert (make_pair (k, v)); k.clear (); state = KEY; } else { v += url[i]; } break; } } if (state == VALUE) { r.insert (make_pair (k, v)); } return r; } string video_asset_filename (shared_ptr asset, int reel_index, int reel_count, optional summary) { dcp::NameFormat::Map values; values['t'] = "j2c"; values['r'] = raw_convert (reel_index + 1); values['n'] = raw_convert (reel_count); if (summary) { values['c'] = careful_string_filter (summary.get()); } return Config::instance()->dcp_asset_filename_format().get(values, "_" + asset->id() + ".mxf"); } string audio_asset_filename (shared_ptr asset, int reel_index, int reel_count, optional summary) { dcp::NameFormat::Map values; values['t'] = "pcm"; values['r'] = raw_convert (reel_index + 1); values['n'] = raw_convert (reel_count); if (summary) { values['c'] = careful_string_filter (summary.get()); } return Config::instance()->dcp_asset_filename_format().get(values, "_" + asset->id() + ".mxf"); } float relaxed_string_to_float (string s) { try { boost::algorithm::replace_all (s, ",", "."); return lexical_cast (s); } catch (bad_lexical_cast &) { boost::algorithm::replace_all (s, ".", ","); return lexical_cast (s); } } string careful_string_filter (string s) { /* Filter out `bad' characters which `may' cause problems with some systems (either for DCP name or filename). There's no apparent list of what really is allowed, so this is a guess. Safety first and all that. */ wstring ws = boost::locale::conv::utf_to_utf(s); string out; string const allowed = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz-_%.+"; for (size_t i = 0; i < ws.size(); ++i) { wchar_t c = ws[i]; /* Remove some accents */ if (wstring(L"áàâ").find(c) != string::npos) { c = 'a'; } if (wstring(L"éèêë").find(c) != string::npos) { c = 'e'; } if (wstring(L"ö").find(c) != string::npos) { c = 'o'; } if (wstring(L"ü").find(c) != string::npos) { c = 'u'; } if (allowed.find(c) != string::npos) { out += c; } } return boost::locale::conv::utf_to_utf(out); } /** @param mapped List of mapped audio channels from a Film. * @param channels Total number of channels in the Film. * @return First: number of non-LFE soundtrack channels (L/R/C/Ls/Rs/Lc/Rc/Bsl/Bsr), second: number of LFE channels. */ pair audio_channel_types (list mapped, int channels) { int non_lfe = 0; int lfe = 0; BOOST_FOREACH (int i, mapped) { if (i >= channels) { /* This channel is mapped but is not included in the DCP */ continue; } switch (static_cast(i)) { case dcp::LFE: ++lfe; break; case dcp::LEFT: case dcp::RIGHT: case dcp::CENTRE: case dcp::LS: case dcp::RS: case dcp::LC: case dcp::RC: case dcp::BSL: case dcp::BSR: ++non_lfe; break; case dcp::HI: case dcp::VI: break; } } return make_pair (non_lfe, lfe); } shared_ptr remap (shared_ptr input, int output_channels, AudioMapping map) { shared_ptr mapped (new AudioBuffers (output_channels, input->frames())); mapped->make_silent (); for (int i = 0; i < map.input_channels(); ++i) { for (int j = 0; j < mapped->channels(); ++j) { if (map.get (i, static_cast (j)) > 0) { mapped->accumulate_channel ( input.get(), i, static_cast (j), map.get (i, static_cast (j)) ); } } } return mapped; } Eyes increment_eyes (Eyes e) { if (e == EYES_LEFT) { return EYES_RIGHT; } return EYES_LEFT; } void checked_fwrite (void const * ptr, size_t size, FILE* stream, boost::filesystem::path path) { size_t N = fwrite (ptr, 1, size, stream); if (N != size) { if (ferror(stream)) { fclose (stream); throw FileError (String::compose("fwrite error %1", errno), path); } else { fclose (stream); throw FileError ("Unexpected short write", path); } } } void checked_fread (void* ptr, size_t size, FILE* stream, boost::filesystem::path path) { size_t N = fread (ptr, 1, size, stream); if (N != size) { if (ferror(stream)) { fclose (stream); throw FileError (String::compose("fread error %1", errno), path); } else { fclose (stream); throw FileError ("Unexpected short read", path); } } } size_t utf8_strlen (string s) { size_t const len = s.length (); int N = 0; for (size_t i = 0; i < len; ++i) { unsigned char c = s[i]; if ((c & 0xe0) == 0xc0) { ++i; } else if ((c & 0xf0) == 0xe0) { i += 2; } else if ((c & 0xf8) == 0xf0) { i += 3; } ++N; } return N; } string day_of_week_to_string (boost::gregorian::greg_weekday d) { switch (d.as_enum()) { case boost::date_time::Sunday: return _("Sunday"); case boost::date_time::Monday: return _("Monday"); case boost::date_time::Tuesday: return _("Tuesday"); case boost::date_time::Wednesday: return _("Wednesday"); case boost::date_time::Thursday: return _("Thursday"); case boost::date_time::Friday: return _("Friday"); case boost::date_time::Saturday: return _("Saturday"); } return d.as_long_string (); } /** @param size Size of picture that the subtitle will be overlaid onto */ void emit_subtitle_image (ContentTimePeriod period, dcp::SubtitleImage sub, dcp::Size size, shared_ptr decoder) { /* XXX: this is rather inefficient; decoding the image just to get its size */ FFmpegImageProxy proxy (sub.png_image()); shared_ptr image = proxy.image().image; /* set up rect with height and width */ dcpomatic::Rect rect(0, 0, image->size().width / double(size.width), image->size().height / double(size.height)); /* add in position */ switch (sub.h_align()) { case dcp::HALIGN_LEFT: rect.x += sub.h_position(); break; case dcp::HALIGN_CENTER: rect.x += 0.5 + sub.h_position() - rect.width / 2; break; case dcp::HALIGN_RIGHT: rect.x += 1 - sub.h_position() - rect.width; break; } switch (sub.v_align()) { case dcp::VALIGN_TOP: rect.y += sub.v_position(); break; case dcp::VALIGN_CENTER: rect.y += 0.5 + sub.v_position() - rect.height / 2; break; case dcp::VALIGN_BOTTOM: rect.y += 1 - sub.v_position() - rect.height; break; } decoder->emit_bitmap (period, image, rect); } bool show_jobs_on_console (bool progress) { bool first = true; bool error = false; while (true) { dcpomatic_sleep_seconds (5); list > jobs = JobManager::instance()->get(); if (!first && progress) { for (size_t i = 0; i < jobs.size(); ++i) { cout << "\033[1A\033[2K"; } cout.flush (); } first = false; BOOST_FOREACH (shared_ptr i, jobs) { if (progress) { cout << i->name(); if (!i->sub_name().empty()) { cout << "; " << i->sub_name(); } cout << ": "; if (i->progress ()) { cout << i->status() << " \n"; } else { cout << ": Running \n"; } } if (!progress && i->finished_in_error()) { /* We won't see this error if we haven't been showing progress, so show it now. */ cout << i->status() << "\n"; } if (i->finished_in_error()) { error = true; } } if (!JobManager::instance()->work_to_do()) { break; } } return error; } /** XXX: could use mmap? */ void copy_in_bits (boost::filesystem::path from, boost::filesystem::path to, boost::function progress) { FILE* f = fopen_boost (from, "rb"); if (!f) { throw OpenFileError (from, errno, OpenFileError::READ); } FILE* t = fopen_boost (to, "wb"); if (!t) { fclose (f); throw OpenFileError (to, errno, OpenFileError::WRITE); } /* on the order of a second's worth of copying */ boost::uintmax_t const chunk = 20 * 1024 * 1024; uint8_t* buffer = static_cast (malloc(chunk)); if (!buffer) { throw std::bad_alloc (); } boost::uintmax_t const total = boost::filesystem::file_size (from); boost::uintmax_t remaining = total; while (remaining) { boost::uintmax_t this_time = min (chunk, remaining); size_t N = fread (buffer, 1, chunk, f); if (N < this_time) { fclose (f); fclose (t); free (buffer); throw ReadFileError (from, errno); } N = fwrite (buffer, 1, this_time, t); if (N < this_time) { fclose (f); fclose (t); free (buffer); throw WriteFileError (to, errno); } progress (1 - float(remaining) / total); remaining -= this_time; } fclose (f); fclose (t); free (buffer); } #ifdef DCPOMATIC_VARIANT_SWAROOP /* Make up a key from the machine UUID */ dcp::Data key_from_uuid () { dcp::Data key (dcpomatic::crypto_key_length()); memset (key.data().get(), 0, key.size()); string const magic = command_and_read ("dcpomatic2_uuid"); strncpy ((char *) key.data().get(), magic.c_str(), dcpomatic::crypto_key_length()); return key; } /* swaroop chain file format: * * 0 [int16_t] IV length * 2 [int16_t] cert #1 length, or 0 for none * 4 [int16_t] cert #2 length, or 0 for none * 6 [int16_t] cert #3 length, or 0 for none * 8 [int16_t] cert #4 length, or 0 for none * 10 [int16_t] cert #5 length, or 0 for none * 12 [int16_t] cert #6 length, or 0 for none * 14 [int16_t] cert #7 length, or 0 for none * 16 [int16_t] cert #8 length, or 0 for none * 16 [int16_t] private key length * 20 IV * cert #1 * cert #2 * cert #3 * cert #4 * cert #5 * cert #6 * cert #7 * cert #8 * private key */ struct __attribute__ ((packed)) Header_ { int16_t iv_length; int16_t cert_length[8]; int16_t private_key_length; }; typedef struct Header_ Header; shared_ptr read_swaroop_chain (boost::filesystem::path path) { dcp::Data data (path); Header* header = (Header *) data.data().get(); uint8_t* p = data.data().get() + sizeof(Header); dcp::Data iv (p, header->iv_length); p += iv.size(); shared_ptr cc (new dcp::CertificateChain()); for (int i = 0; i < 8; ++i) { if (header->cert_length[i] == 0) { break; } dcp::Data c(p, header->cert_length[i]); p += c.size(); cc->add (dcp::Certificate(dcpomatic::decrypt(c, key_from_uuid(), iv))); } dcp::Data k (p, header->private_key_length); cc->set_key (dcpomatic::decrypt(k, key_from_uuid(), iv)); return cc; } void write_swaroop_chain (shared_ptr chain, boost::filesystem::path output) { scoped_array buffer (new uint8_t[65536]); Header* header = (Header *) buffer.get(); memset (header, 0, sizeof(Header)); uint8_t* p = buffer.get() + sizeof(Header); dcp::Data iv = dcpomatic::random_iv (); header->iv_length = iv.size (); memcpy (p, iv.data().get(), iv.size()); p += iv.size(); int N = 0; BOOST_FOREACH (dcp::Certificate i, chain->root_to_leaf()) { dcp::Data e = dcpomatic::encrypt (i.certificate(true), key_from_uuid(), iv); memcpy (p, e.data().get(), e.size()); p += e.size(); DCPOMATIC_ASSERT (N < 8); header->cert_length[N] = e.size (); ++N; } dcp::Data k = dcpomatic::encrypt (chain->key().get(), key_from_uuid(), iv); memcpy (p, k.data().get(), k.size()); p += k.size(); header->private_key_length = k.size (); FILE* f = fopen_boost (output, "wb"); checked_fwrite (buffer.get(), p - buffer.get(), f, output); fclose (f); } #endif double db_to_linear (double db) { return pow(10, db / 20); } double linear_to_db (double linear) { return 20 * log10(linear); }