forked from tsukumijima/rplsinfo
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrplsinfo.cpp
514 lines (427 loc) · 27.3 KB
/
rplsinfo.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
// rplsinfo.cpp : コンソール アプリケーションのエントリ ポイントを定義します。
//
#include "stdafx.h"
#include <windows.h>
#include <stdio.h>
#include <string.h>
#include <new.h>
#include <locale.h>
#include <wchar.h>
#include "rplsinfo.h"
#include "tsproginfo.h"
#include "rplsproginfo.h"
#include "tsprocess.h"
#include "convToUnicode.h"
#include "conio.h"
// 定数など
#ifdef _WIN64
#define NAMESTRING "\nrplsinfo version 1.5.2 (64bit)\n"
#else
#define NAMESTRING "\nrplsinfo version 1.5.2 (32bit)\n"
#endif
// マクロ定義
#define printMsg(fmt, ...) _tprintf(_T(fmt), __VA_ARGS__)
#define printErrMsg(fmt, ...) _tprintf(_T(fmt), __VA_ARGS__)
// プロトタイプ宣言
void initCopyParams(CopyParams*);
bool parseCopyParams(const int32_t, _TCHAR *[], CopyParams *);
size_t convForCsv(WCHAR*, const size_t, const WCHAR*, const size_t, const CopyParams*);
void outputProgInfo(HANDLE, const ProgInfo*, const CopyParams*);
//
int _tmain(int argc, _TCHAR* argv[])
{
_tsetlocale(LC_ALL, _T(""));
// パラメータチェック
if (argc == 1) {
printMsg(NAMESTRING);
exit(1);
}
CopyParams param;
initCopyParams(¶m);
if (!parseCopyParams(argc, argv, ¶m)) exit(1); // パラメータ異常なら終了
// 番組情報元ファイルから番組情報を読み込む
ProgInfo proginfo;
memset(&proginfo, 0, sizeof(proginfo));
if (!readFileProgInfo(argv[param.argSrc], &proginfo, ¶m)) exit(1); // 失敗したら終了する
// 必要なら出力ファイルを開く
uint32_t numWrite;
HANDLE hWriteFile = INVALID_HANDLE_VALUE;
if (!param.bDisplay)
{
hWriteFile = CreateFile(argv[param.argDest], GENERIC_WRITE | GENERIC_READ, 0, NULL, OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
if (hWriteFile == INVALID_HANDLE_VALUE) {
printErrMsg("保存先ファイル %s を開けませんでした.\n", argv[param.argDest]);
exit(1);
}
int64_t fsize;
fsize = GetFileDataSize(hWriteFile); // ファイルサイズ取得
if (param.bUseBOM && (fsize == 0)) {
const WCHAR bom = 0xFEFF;
WriteFileData(hWriteFile, (uint8_t*)&bom, sizeof(bom), &numWrite); // ファイルが空ならBOM出力
}
else {
SeekFileData(hWriteFile, fsize); // 空でないならファイルの最後尾に移動
}
}
// オプション指定に応じて番組内容を出力
outputProgInfo(hWriteFile, &proginfo, ¶m);
// 終了処理
if (!param.bDisplay) CloseHandle(hWriteFile);
return 0;
}
void initCopyParams(CopyParams* param)
{
param->argSrc = 0;
param->argDest = 0;
param->separator = S_NORMAL;
param->bNoControl = true;
param->bNoComma = true;
param->bDQuot = false;
param->bItemName = false;
param->bDisplay = false;
param->bCharSize = false;
param->bIVS = false;
param->bUseBOM = true;
param->packet_limit = 0;
param->tsfilepos = DEFAULTTSFILEPOS;
memset(param->flags, 0, sizeof(param->flags));
return;
}
bool parseCopyParams(const int32_t argn, _TCHAR *args[], CopyParams *param)
{
int32_t fnum = 0;
bool bArgSkip = false;
for (int32_t i = 1; i < argn; i++)
{
if (args[i][0] == L'-')
{
for (int32_t j = 1; j < (int32_t)wcslen(args[i]); j++)
{
switch (args[i][j])
{
case L'f':
param->flags[fnum++] = F_FileName;
break;
case L'u':
param->flags[fnum++] = F_FileNameFullPath;
break;
case L'k':
param->flags[fnum++] = F_FileSize;
break;
case L'd':
param->flags[fnum++] = F_RecDate;
break;
case L't':
param->flags[fnum++] = F_RecTime;
break;
case L'p':
param->flags[fnum++] = F_RecDuration;
break;
case L'z':
param->flags[fnum++] = F_RecTimeZone;
break;
case L'a':
param->flags[fnum++] = F_MakerID;
break;
case L'o':
param->flags[fnum++] = F_ModelCode;
break;
case L's':
param->flags[fnum++] = F_RecSrc;
break;
case L'c':
param->flags[fnum++] = F_ChannelName;
break;
case L'n':
param->flags[fnum++] = F_ChannelNum;
break;
case L'b':
param->flags[fnum++] = F_ProgName;
break;
case L'i':
param->flags[fnum++] = F_ProgDetail;
break;
case L'e':
param->flags[fnum++] = F_ProgExtend;
break;
case L'g':
param->flags[fnum++] = F_ProgGenre;
break;
case L'v':
param->flags[fnum++] = F_ProgVideo;
break;
case L'm':
param->flags[fnum++] = F_ProgAudio;
break;
case L'T':
param->separator = S_TAB;
param->bNoControl = true;
param->bNoComma = false;
param->bDQuot = false;
param->bItemName = false;
break;
case L'S':
param->separator = S_SPACE;
param->bNoControl = true;
param->bNoComma = false;
param->bDQuot = false;
param->bItemName = false;
break;
case L'C':
param->separator = S_CSV;
param->bNoControl = false;
param->bNoComma = false;
param->bDQuot = true;
param->bItemName = false;
break;
case L'N':
param->separator = S_NEWLINE;
param->bNoControl = false;
param->bNoComma = false;
param->bDQuot = false;
param->bItemName = false;
break;
case L'I':
param->separator = S_ITEMNAME;
param->bNoControl = false;
param->bNoComma = false;
param->bDQuot = false;
param->bItemName = true;
break;
case L'y':
param->bCharSize = true;
break;
case L'j':
param->bIVS = true;
break;
case L'q':
param->bUseBOM = false;
break;
case L'F':
if ((i == (argn - 1)) || (_wtoi(args[i + 1]) < 0) || (_wtoi(args[i + 1]) > 99)) {
printErrMsg("-F オプションの引数が異常です.\n");
return false;
}
param->tsfilepos = _wtoi(args[i + 1]);
bArgSkip = true;
break;
case L'l':
if ((i == (argn - 1)) || (_wtoi(args[i + 1]) <= 0)) {
printErrMsg("-l オプションの引数が異常です.\n");
return false;
}
param->packet_limit = (int64_t)_wtoi(args[i + 1]) * 5600;
bArgSkip = true;
break;
default:
printErrMsg("無効なスイッチが指定されました.\n");
return false;
}
if (fnum == MAXFLAGNUM) {
printErrMsg("スイッチ指定が多すぎます.\n");
return false;
}
}
}
else {
if (param->argSrc == 0) {
param->argSrc = i;
}
else if (param->argDest == 0) {
param->argDest = i;
}
else {
printErrMsg("パラメータが多すぎます.\n");
return false;
}
}
if (bArgSkip) {
i++;
bArgSkip = false;
}
}
if (param->argSrc == 0) {
printErrMsg("パラメータが足りません.\n");
return false;
}
if (param->argDest == 0) param->bDisplay = true;
return true;
}
size_t convForCsv(WCHAR* dbuf, const size_t bufsize, const WCHAR* sbuf, const size_t slen, const CopyParams* param)
{
size_t dst = 0;
if (param->bDQuot && (dst < bufsize)) dbuf[dst++] = 0x0022; // 「"」 // CSV用出力なら項目の前後を「"」で囲む
for (size_t src = 0; src < slen; src++)
{
const WCHAR s = sbuf[src];
bool bOutput = true;
if (param->bNoControl && (s < L' ')) bOutput = false; // 制御コードは出力しない
if (param->bNoComma && (s == L',')) bOutput = false; // コンマは出力しない
if (param->bDisplay && (s == 0x000D)) bOutput = false; // コンソール出力の場合は改行コードの0x000Dは省略する
if (param->bDQuot && (s == 0x0022) && (dst < bufsize)) dbuf[dst++] = 0x0022; // CSV用出力なら「"」の前に「"」でエスケープ
if (bOutput && (dst < bufsize)) dbuf[dst++] = s; // 出力
}
if (param->bDQuot && (dst < bufsize)) dbuf[dst++] = 0x0022; // 「"」
if (dst < bufsize) dbuf[dst] = 0x0000;
return dst;
}
void outputProgInfo(HANDLE hFile, const ProgInfo* proginfo, const CopyParams* param)
{
WCHAR sstr[CONVBUFSIZE];
WCHAR dstr[CONVBUFSIZE];
uint32_t numWrite;
int32_t i = 0;
while (param->flags[i] != 0) {
size_t slen = 0;
// flagsに応じて出力する項目をsstrにコピーする
switch (param->flags[i])
{
case F_FileName:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[ファイル名]\r\n");
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"%s%s", proginfo->fname, proginfo->fext);
break;
case F_FileNameFullPath:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[フルパスファイル名]\r\n");
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"%s", proginfo->fullpath);
break;
case F_FileSize:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[ファイルサイズ]\r\n");
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"%lld", proginfo->fsize);
break;
case F_RecDate:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[録画日付]\r\n");
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"%.4d/%.2d/%.2d", proginfo->recyear, proginfo->recmonth, proginfo->recday);
break;
case F_RecTime:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[録画時刻]\r\n");
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"%.2d:%.2d:%.2d", proginfo->rechour, proginfo->recmin, proginfo->recsec);
break;
case F_RecDuration:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[録画期間]\r\n");
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"%.2d:%.2d:%.2d", proginfo->durhour, proginfo->durmin, proginfo->dursec);
break;
case F_RecTimeZone:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[タイムゾーン]\r\n");
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"%d", proginfo->rectimezone);
break;
case F_MakerID:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[メーカーID]\r\n");
if (proginfo->makerid != -1) {
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"%d", proginfo->makerid);
}
else {
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"n/a");
}
break;
case F_ModelCode:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[メーカー機種コード]\r\n");
if (proginfo->modelcode != -1) {
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"%d", proginfo->modelcode);
}
else {
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"n/a");
}
break;
case F_RecSrc:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[放送種別]\r\n");
slen += getRecSrcStr(sstr + slen, CONVBUFSIZE - slen, proginfo->recsrc);
break;
case F_ChannelNum:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[チャンネル番号]\r\n");
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"%.3dch", proginfo->chnum);
break;
case F_ChannelName:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[放送局名]\r\n");
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"%s", proginfo->chname);
break;
case F_ProgName:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[番組名]\r\n");
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"%s", proginfo->pname);
break;
case F_ProgDetail:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[番組内容]\r\n");
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"%s", proginfo->pdetail);
break;
case F_ProgExtend:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[内容詳細]\r\n");
if (proginfo->pextendlen == -1) {
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"n/a");
}
else {
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"%s", proginfo->pextend);
}
break;
case F_ProgGenre:
if (param->bItemName) slen = swprintf_s(sstr, CONVBUFSIZE, L"[番組ジャンル]\r\n");
slen += putGenreStr(sstr + slen, CONVBUFSIZE - slen, proginfo->genre);
break;
case F_ProgVideo:
if (param->bItemName) slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"[映像]\r\n");
slen += putFormatStr(sstr + slen, CONVBUFSIZE - slen, proginfo->videoformat);
break;
case F_ProgAudio:
if (param->bItemName) slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"[音声]\r\n");
for (int32_t audioNum = 0; audioNum < 8; audioNum++)
{
if (proginfo->audioformat[audioNum] == -1) break;
if (audioNum != 0) slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L"\r\n");
slen += putFormatStr(sstr + slen, CONVBUFSIZE - slen, proginfo->audioformat[audioNum]);
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L" ");
slen += putSamplingrateStr(sstr + slen, CONVBUFSIZE - slen, proginfo->audiosamplingrate[audioNum]);
if (proginfo->audiotextlen[audioNum] != 0) slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L" %s", proginfo->audiotext[audioNum]);
slen += swprintf_s(sstr + slen, CONVBUFSIZE - slen, L" (%hs)", proginfo->audiolang[audioNum]);
}
break;
default:
slen += 0;
break;
}
// 出力形式に応じてsstrを調整
size_t dlen = convForCsv(dstr, CONVBUFSIZE - 5, sstr, slen, param);
// セパレータに関する処理
if (param->flags[i + 1] != 0)
{
switch (param->separator)
{
case S_NORMAL:
case S_CSV:
dstr[dlen++] = L','; // COMMA
break;
case S_TAB:
dstr[dlen++] = 0x0009; // TAB
break;
case S_SPACE:
dstr[dlen++] = L' '; // SPACE
break;
case S_NEWLINE:
case S_ITEMNAME:
if (!param->bDisplay) dstr[dlen++] = 0x000D; // 改行2回
dstr[dlen++] = 0x000A;
if (!param->bDisplay) dstr[dlen++] = 0x000D;
dstr[dlen++] = 0x000A;
break;
default:
break;
}
}
else
{
if (!param->bDisplay) dstr[dlen++] = 0x000D; // 全項目出力後の改行
dstr[dlen++] = 0x000A;
if ((param->separator == S_NEWLINE) || (param->separator == S_ITEMNAME)) { // セパレータがS_NEWLINE, S_ITEMNAMEの場合は1行余分に改行する
if (!param->bDisplay) dstr[dlen++] = 0x000D;
dstr[dlen++] = 0x000A;
}
}
// データ出力
if (!param->bDisplay) {
WriteFileData(hFile, (uint8_t*)dstr, (uint32_t)dlen * sizeof(WCHAR), &numWrite); // ディスク出力
}
else {
dstr[dlen] = 0x0000;
printMsg("%s", dstr); // コンソール出力
}
i++;
}
return;
}