mpegts.cc 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375
  1. /*
  2. * Copyright (C) 2018-2020 Stefan Westerfeld
  3. *
  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 3 of the License, or
  7. * (at your option) any later version.
  8. *
  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.
  13. *
  14. * You should have received a copy of the GNU General Public License
  15. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  16. */
  17. #include <array>
  18. #include <regex>
  19. #include <string.h>
  20. #include "utils.hh"
  21. #include "mpegts.hh"
  22. using std::string;
  23. using std::vector;
  24. using std::map;
  25. using std::regex;
  26. class TSPacket
  27. {
  28. public:
  29. enum class ID { awmk_file, awmk_data, unknown };
  30. private:
  31. std::array<unsigned char, 188> m_data;
  32. std::array<unsigned char, 12>
  33. get_id_bytes (ID type)
  34. {
  35. if (type == ID::awmk_file)
  36. return { 'G', 0x1F, 0xFF, 0x10, 'A', 'W', 'M', 'K', 'f', 'i', 'l', 'e' };
  37. if (type == ID::awmk_data)
  38. return { 'G', 0x1F, 0xFF, 0x10, 'A', 'W', 'M', 'K', 'd', 'a', 't', 'a' };
  39. return {0,};
  40. }
  41. public:
  42. bool
  43. read (FILE *file, Error& err)
  44. {
  45. size_t bytes_read = fread (m_data.data(), 1, m_data.size(), file);
  46. if (bytes_read == 0) /* probably eof */
  47. return false;
  48. if (bytes_read == m_data.size()) /* successful read */
  49. {
  50. if (m_data[0] == 'G')
  51. return true;
  52. err = Error ("bad packet sync while reading transport (.ts) packet");
  53. return false;
  54. }
  55. err = Error ("short read while reading transport stream (.ts) packet");
  56. return false;
  57. }
  58. Error
  59. write (FILE *file)
  60. {
  61. size_t bytes_written = fwrite (m_data.data(), 1, m_data.size(), file);
  62. if (bytes_written != m_data.size())
  63. return Error ("short write while writing transport stream (.ts) packet");
  64. return Error::Code::NONE;
  65. }
  66. void
  67. clear (ID type)
  68. {
  69. std::fill (m_data.begin(), m_data.end(), 0);
  70. auto id = get_id_bytes (type);
  71. std::copy (id.begin(), id.end(), m_data.begin());
  72. }
  73. unsigned char&
  74. operator[] (size_t n)
  75. {
  76. return m_data[n];
  77. }
  78. bool
  79. id_eq (size_t offset, unsigned char a, unsigned char b, unsigned char c, unsigned char d)
  80. {
  81. return m_data[offset] == a && m_data[offset + 1] == b && m_data[offset + 2] == c && m_data[offset + 3] == d;
  82. }
  83. ID
  84. get_id()
  85. {
  86. if (id_eq (0, 'G', 0x1F, 0xFF, 0x10) && id_eq (4, 'A', 'W', 'M', 'K'))
  87. {
  88. if (id_eq (8, 'f', 'i', 'l', 'e'))
  89. return ID::awmk_file;
  90. if (id_eq (8, 'd', 'a', 't', 'a'))
  91. return ID::awmk_data;
  92. }
  93. return ID::unknown;
  94. }
  95. constexpr size_t
  96. size()
  97. {
  98. return m_data.size(); // is constant
  99. }
  100. const std::array<unsigned char, 188>&
  101. data()
  102. {
  103. return m_data;
  104. }
  105. };
  106. Error
  107. TSWriter::append_file (const string& name, const string& filename)
  108. {
  109. vector<unsigned char> data;
  110. FILE *datafile = fopen (filename.c_str(), "r");
  111. ScopedFile datafile_s (datafile);
  112. if (!datafile)
  113. return Error ("unable to open data file");
  114. int c;
  115. while ((c = fgetc (datafile)) >= 0)
  116. data.push_back (c);
  117. entries.push_back ({name, data});
  118. return Error::Code::NONE;
  119. }
  120. void
  121. TSWriter::append_vars (const string& name, const map<string, string>& vars)
  122. {
  123. vector<unsigned char> data;
  124. for (auto kv : vars)
  125. {
  126. for (auto k : kv.first)
  127. data.push_back (k);
  128. data.push_back ('=');
  129. for (auto v : kv.second)
  130. data.push_back (v);
  131. data.push_back (0);
  132. }
  133. entries.push_back ({name, data});
  134. }
  135. void
  136. TSWriter::append_data (const string& name, const vector<unsigned char>& data)
  137. {
  138. entries.push_back ({name, data});
  139. }
  140. Error
  141. TSWriter::process (const string& inname, const string& outname)
  142. {
  143. FILE *infile = fopen (inname.c_str(), "r");
  144. FILE *outfile = fopen (outname.c_str(), "w");
  145. ScopedFile infile_s (infile);
  146. ScopedFile outfile_s (outfile);
  147. if (!infile)
  148. {
  149. error ("audiowmark: unable to open %s for reading\n", inname.c_str());
  150. return Error (strerror (errno));
  151. }
  152. if (!outfile)
  153. {
  154. error ("audiowmark: unable to open %s for writing\n", outname.c_str());
  155. return Error (strerror (errno));
  156. }
  157. while (!feof (infile))
  158. {
  159. TSPacket p;
  160. Error err;
  161. bool read_ok = p.read (infile, err);
  162. if (!read_ok)
  163. {
  164. if (err)
  165. return err;
  166. }
  167. else
  168. {
  169. err = p.write (outfile);
  170. if (err)
  171. return err;
  172. }
  173. }
  174. for (auto entry : entries)
  175. {
  176. string header = string_printf ("%zd:%s", entry.data.size(), entry.name.c_str()) + '\0';
  177. vector<unsigned char> data = entry.data;
  178. for (size_t i = 0; i < header.size(); i++)
  179. data.insert (data.begin() + i, header[i]);
  180. TSPacket p_file;
  181. p_file.clear (TSPacket::ID::awmk_file);
  182. size_t data_pos = 0;
  183. int pos = 12;
  184. while (data_pos < data.size())
  185. {
  186. p_file[pos++] = data[data_pos];
  187. if (pos == 188)
  188. {
  189. Error err = p_file.write (outfile);
  190. if (err)
  191. return err;
  192. p_file.clear (TSPacket::ID::awmk_data);
  193. pos = 12;
  194. }
  195. data_pos++;
  196. }
  197. if (pos != 12)
  198. {
  199. Error err = p_file.write (outfile);
  200. if (err)
  201. return err;
  202. }
  203. }
  204. return Error::Code::NONE;
  205. }
  206. bool
  207. TSReader::parse_header (Header& header, vector<unsigned char>& data)
  208. {
  209. for (size_t i = 0; i < data.size(); i++)
  210. {
  211. if (data[i] == 0) // header is terminated with one single 0 byte
  212. {
  213. string s = (const char *) (&data[0]);
  214. static const regex header_re ("([0-9]*):(.*)");
  215. std::smatch sm;
  216. if (regex_match (s, sm, header_re))
  217. {
  218. header.data_size = atoi (sm[1].str().c_str());
  219. header.filename = sm[2];
  220. // erase header including null termination
  221. data.erase (data.begin(), data.begin() + i + 1);
  222. return true;
  223. }
  224. }
  225. }
  226. return false;
  227. }
  228. Error
  229. TSReader::load (const string& inname)
  230. {
  231. if (inname == "-")
  232. {
  233. return load (stdin);
  234. }
  235. else
  236. {
  237. FILE *infile = fopen (inname.c_str(), "r");
  238. ScopedFile infile_s (infile);
  239. if (!infile)
  240. return Error (string_printf ("error opening input .ts '%s'", inname.c_str()));
  241. return load (infile);
  242. }
  243. }
  244. Error
  245. TSReader::load (FILE *infile)
  246. {
  247. vector<unsigned char> awmk_stream;
  248. Header header;
  249. bool header_valid = false;
  250. Error err;
  251. while (!feof (infile))
  252. {
  253. TSPacket p;
  254. bool read_ok = p.read (infile, err);
  255. if (!read_ok)
  256. {
  257. if (err)
  258. return err;
  259. }
  260. else
  261. {
  262. TSPacket::ID id = p.get_id();
  263. if (id == TSPacket::ID::awmk_file)
  264. {
  265. /* new stream start, clear old contents */
  266. header_valid = false;
  267. awmk_stream.clear();
  268. }
  269. if (id == TSPacket::ID::awmk_file || id == TSPacket::ID::awmk_data)
  270. {
  271. awmk_stream.insert (awmk_stream.end(), p.data().begin() + 12, p.data().end());
  272. if (!header_valid)
  273. {
  274. if (parse_header (header, awmk_stream))
  275. {
  276. awmk_stream.reserve (header.data_size + p.size());
  277. header_valid = true;
  278. }
  279. }
  280. // done? do we have enough bytes for the complete entry?
  281. if (header_valid && awmk_stream.size() >= header.data_size)
  282. {
  283. awmk_stream.resize (header.data_size);
  284. m_entries.push_back ({ header.filename, std::move (awmk_stream)});
  285. header_valid = false;
  286. awmk_stream.clear();
  287. }
  288. }
  289. }
  290. }
  291. return Error::Code::NONE;
  292. }
  293. const vector<TSReader::Entry>&
  294. TSReader::entries()
  295. {
  296. return m_entries;
  297. }
  298. const TSReader::Entry *
  299. TSReader::find (const string& name) const
  300. {
  301. for (const auto& entry : m_entries)
  302. if (entry.filename == name)
  303. return &entry;
  304. return nullptr;
  305. }
  306. map<string, string>
  307. TSReader::parse_vars (const string& name)
  308. {
  309. map<string, string> vars;
  310. auto entry = find (name);
  311. if (!entry)
  312. return vars;
  313. enum { KEY, VALUE } mode = KEY;
  314. string s;
  315. string key;
  316. for (auto c : entry->data)
  317. {
  318. if (c == '=' && mode == KEY)
  319. {
  320. key = s;
  321. s.clear();
  322. mode = VALUE;
  323. }
  324. else if (c == '\0' && mode == VALUE)
  325. {
  326. vars[key] = s;
  327. s.clear();
  328. mode = KEY;
  329. }
  330. else
  331. {
  332. s += c;
  333. }
  334. }
  335. return vars;
  336. }