Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Contribute to GitLab
Sign in / Register
Toggle navigation
F
ffmpeg.wasm-core
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Linshizhi
ffmpeg.wasm-core
Commits
cbba331a
Commit
cbba331a
authored
Oct 02, 2013
by
Stefano Sabatini
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
ffprobe: implement string validation setting
This should fix trac tickets #1163, #2502.
parent
11cba3ba
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
180 additions
and
12 deletions
+180
-12
Changelog
Changelog
+1
-0
ffprobe.texi
doc/ffprobe.texi
+33
-0
ffprobe.c
ffprobe.c
+146
-12
No files found.
Changelog
View file @
cbba331a
...
@@ -7,6 +7,7 @@ version <next>
...
@@ -7,6 +7,7 @@ version <next>
- Live HDS muxer
- Live HDS muxer
- setsar/setdar filters now support variables in ratio expressions
- setsar/setdar filters now support variables in ratio expressions
- elbg filter
- elbg filter
- string validation in ffprobe
version 2.1:
version 2.1:
...
...
doc/ffprobe.texi
View file @
cbba331a
...
@@ -337,6 +337,39 @@ A writer may accept one or more arguments, which specify the options
...
@@ -337,6 +337,39 @@ A writer may accept one or more arguments, which specify the options
to adopt. The options are specified as a list of @var
{
key
}
=@var
{
value
}
to adopt. The options are specified as a list of @var
{
key
}
=@var
{
value
}
pairs, separated by ":".
pairs, separated by ":".
All writers support the following options:
@table @option
@item string
_
validation, sv
Set string validation mode.
The following values are accepted.
@table @samp
@item fail
The writer will fail immediately in case an invalid string (UTF-8)
sequence or code point is found in the input. This is especially
useful to validate input metadata.
@item ignore
Any validation error will be ignored. This will result in possibly
broken output, especially with the json or xml writer.
@item replace
The writer will substitute invalid UTF-8 sequences or code points with
the string specified with the @option
{
string
_
validation
_
replacement
}
.
@end table
Default value is @samp
{
replace
}
.
@item string
_
validation
_
replacement, svr
Set replacement string to use in case @option
{
string
_
validation
}
is
set to @samp
{
replace
}
.
In case the option is not specified, the writer will assume the empty
string, that is it will remove the invalid sequences from the input
strings.
@end table
A description of the currently available writers follows.
A description of the currently available writers follows.
@section default
@section default
...
...
ffprobe.c
View file @
cbba331a
...
@@ -258,6 +258,13 @@ typedef struct WriterContext WriterContext;
...
@@ -258,6 +258,13 @@ typedef struct WriterContext WriterContext;
#define WRITER_FLAG_DISPLAY_OPTIONAL_FIELDS 1
#define WRITER_FLAG_DISPLAY_OPTIONAL_FIELDS 1
#define WRITER_FLAG_PUT_PACKETS_AND_FRAMES_IN_SAME_CHAPTER 2
#define WRITER_FLAG_PUT_PACKETS_AND_FRAMES_IN_SAME_CHAPTER 2
typedef
enum
{
WRITER_STRING_VALIDATION_FAIL
,
WRITER_STRING_VALIDATION_REPLACE
,
WRITER_STRING_VALIDATION_IGNORE
,
WRITER_STRING_VALIDATION_NB
,
}
StringValidation
;
typedef
struct
Writer
{
typedef
struct
Writer
{
const
AVClass
*
priv_class
;
///< private class of the writer, if any
const
AVClass
*
priv_class
;
///< private class of the writer, if any
int
priv_size
;
///< private size for the writer context
int
priv_size
;
///< private size for the writer context
...
@@ -298,6 +305,10 @@ struct WriterContext {
...
@@ -298,6 +305,10 @@ struct WriterContext {
unsigned
int
nb_section_packet
;
///< number of the packet section in case we are in "packets_and_frames" section
unsigned
int
nb_section_packet
;
///< number of the packet section in case we are in "packets_and_frames" section
unsigned
int
nb_section_frame
;
///< number of the frame section in case we are in "packets_and_frames" section
unsigned
int
nb_section_frame
;
///< number of the frame section in case we are in "packets_and_frames" section
unsigned
int
nb_section_packet_frame
;
///< nb_section_packet or nb_section_frame according if is_packets_and_frames
unsigned
int
nb_section_packet_frame
;
///< nb_section_packet or nb_section_frame according if is_packets_and_frames
StringValidation
string_validation
;
char
*
string_validation_replacement
;
unsigned
int
string_validation_utf8_flags
;
};
};
static
const
char
*
writer_get_name
(
void
*
p
)
static
const
char
*
writer_get_name
(
void
*
p
)
...
@@ -308,6 +319,19 @@ static const char *writer_get_name(void *p)
...
@@ -308,6 +319,19 @@ static const char *writer_get_name(void *p)
#define OFFSET(x) offsetof(WriterContext, x)
#define OFFSET(x) offsetof(WriterContext, x)
static
const
AVOption
writer_options
[]
=
{
{
"string_validation"
,
"set string validation mode"
,
OFFSET
(
string_validation
),
AV_OPT_TYPE_INT
,
{.
i64
=
WRITER_STRING_VALIDATION_REPLACE
},
0
,
WRITER_STRING_VALIDATION_NB
-
1
,
.
unit
=
"sv"
},
{
"sv"
,
"set string validation mode"
,
OFFSET
(
string_validation
),
AV_OPT_TYPE_INT
,
{.
i64
=
WRITER_STRING_VALIDATION_REPLACE
},
0
,
WRITER_STRING_VALIDATION_NB
-
1
,
.
unit
=
"sv"
},
{
"ignore"
,
NULL
,
0
,
AV_OPT_TYPE_CONST
,
{.
i64
=
WRITER_STRING_VALIDATION_IGNORE
},
.
unit
=
"sv"
},
{
"replace"
,
NULL
,
0
,
AV_OPT_TYPE_CONST
,
{.
i64
=
WRITER_STRING_VALIDATION_REPLACE
},
.
unit
=
"sv"
},
{
"fail"
,
NULL
,
0
,
AV_OPT_TYPE_CONST
,
{.
i64
=
WRITER_STRING_VALIDATION_FAIL
},
.
unit
=
"sv"
},
{
"string_validation_replacement"
,
"set string validation replacement string"
,
OFFSET
(
string_validation_replacement
),
AV_OPT_TYPE_STRING
,
{.
str
=
""
}},
{
"svr"
,
"set string validation replacement string"
,
OFFSET
(
string_validation_replacement
),
AV_OPT_TYPE_STRING
,
{.
str
=
""
}},
{
NULL
},
};
static
void
*
writer_child_next
(
void
*
obj
,
void
*
prev
)
static
void
*
writer_child_next
(
void
*
obj
,
void
*
prev
)
{
{
WriterContext
*
ctx
=
obj
;
WriterContext
*
ctx
=
obj
;
...
@@ -321,6 +345,7 @@ static const AVClass writer_class = {
...
@@ -321,6 +345,7 @@ static const AVClass writer_class = {
writer_get_name
,
writer_get_name
,
NULL
,
NULL
,
LIBAVUTIL_VERSION_INT
,
LIBAVUTIL_VERSION_INT
,
.
option
=
writer_options
,
.
child_next
=
writer_child_next
,
.
child_next
=
writer_child_next
,
};
};
...
@@ -341,6 +366,15 @@ static void writer_close(WriterContext **wctx)
...
@@ -341,6 +366,15 @@ static void writer_close(WriterContext **wctx)
av_freep
(
wctx
);
av_freep
(
wctx
);
}
}
static
void
bprint_bytes
(
AVBPrint
*
bp
,
const
uint8_t
*
ubuf
,
size_t
ubuf_size
)
{
int
i
;
av_bprintf
(
bp
,
"0X"
);
for
(
i
=
0
;
i
<
ubuf_size
;
i
++
)
av_bprintf
(
bp
,
"%02X"
,
ubuf
[
i
]);
}
static
int
writer_open
(
WriterContext
**
wctx
,
const
Writer
*
writer
,
const
char
*
args
,
static
int
writer_open
(
WriterContext
**
wctx
,
const
Writer
*
writer
,
const
char
*
args
,
const
struct
section
*
sections
,
int
nb_sections
)
const
struct
section
*
sections
,
int
nb_sections
)
{
{
...
@@ -393,6 +427,26 @@ static int writer_open(WriterContext **wctx, const Writer *writer, const char *a
...
@@ -393,6 +427,26 @@ static int writer_open(WriterContext **wctx, const Writer *writer, const char *a
av_dict_free
(
&
opts
);
av_dict_free
(
&
opts
);
}
}
/* validate replace string */
{
const
uint8_t
*
p
=
(
*
wctx
)
->
string_validation_replacement
;
const
uint8_t
*
endp
=
p
+
strlen
(
p
);
while
(
*
p
)
{
const
uint8_t
*
p0
=
p
;
int32_t
code
;
ret
=
av_utf8_decode
(
&
code
,
&
p
,
endp
,
(
*
wctx
)
->
string_validation_utf8_flags
);
if
(
ret
<
0
)
{
AVBPrint
bp
;
av_bprint_init
(
&
bp
,
0
,
AV_BPRINT_SIZE_AUTOMATIC
);
bprint_bytes
(
&
bp
,
p0
,
p
-
p0
),
av_log
(
wctx
,
AV_LOG_ERROR
,
"Invalid UTF8 sequence %s found in string validation replace '%s'
\n
"
,
bp
.
str
,
(
*
wctx
)
->
string_validation_replacement
);
return
ret
;
}
}
}
for
(
i
=
0
;
i
<
SECTION_MAX_NB_LEVELS
;
i
++
)
for
(
i
=
0
;
i
<
SECTION_MAX_NB_LEVELS
;
i
++
)
av_bprint_init
(
&
(
*
wctx
)
->
section_pbuf
[
i
],
1
,
AV_BPRINT_SIZE_UNLIMITED
);
av_bprint_init
(
&
(
*
wctx
)
->
section_pbuf
[
i
],
1
,
AV_BPRINT_SIZE_UNLIMITED
);
...
@@ -460,17 +514,94 @@ static inline void writer_print_integer(WriterContext *wctx,
...
@@ -460,17 +514,94 @@ static inline void writer_print_integer(WriterContext *wctx,
}
}
}
}
static
inline
int
validate_string
(
WriterContext
*
wctx
,
char
**
dstp
,
const
char
*
src
)
{
const
uint8_t
*
p
,
*
endp
;
AVBPrint
dstbuf
;
int
invalid_chars_nb
=
0
,
ret
=
0
;
av_bprint_init
(
&
dstbuf
,
0
,
AV_BPRINT_SIZE_UNLIMITED
);
endp
=
src
+
strlen
(
src
);
for
(
p
=
(
uint8_t
*
)
src
;
*
p
;)
{
uint32_t
code
;
int
invalid
=
0
;
const
uint8_t
*
p0
=
p
;
if
(
av_utf8_decode
(
&
code
,
&
p
,
endp
,
wctx
->
string_validation_utf8_flags
)
<
0
)
{
AVBPrint
bp
;
av_bprint_init
(
&
bp
,
0
,
AV_BPRINT_SIZE_AUTOMATIC
);
bprint_bytes
(
&
bp
,
p0
,
p
-
p0
);
av_log
(
wctx
,
AV_LOG_DEBUG
,
"Invalid UTF-8 sequence %s found in string '%s'
\n
"
,
bp
.
str
,
src
);
invalid
=
1
;
}
if
(
invalid
)
{
invalid_chars_nb
++
;
switch
(
wctx
->
string_validation
)
{
case
WRITER_STRING_VALIDATION_FAIL
:
av_log
(
wctx
,
AV_LOG_ERROR
,
"Invalid UTF-8 sequence found in string '%s'
\n
"
,
src
);
ret
=
AVERROR_INVALIDDATA
;
goto
end
;
break
;
case
WRITER_STRING_VALIDATION_REPLACE
:
av_bprintf
(
&
dstbuf
,
"%s"
,
wctx
->
string_validation_replacement
);
break
;
}
}
if
(
!
invalid
||
wctx
->
string_validation
==
WRITER_STRING_VALIDATION_IGNORE
)
av_bprint_append_data
(
&
dstbuf
,
p0
,
p
-
p0
);
}
if
(
invalid_chars_nb
&&
wctx
->
string_validation
==
WRITER_STRING_VALIDATION_REPLACE
)
{
av_log
(
wctx
,
AV_LOG_WARNING
,
"%d invalid UTF-8 sequence(s) found in string '%s', replaced with '%s'
\n
"
,
invalid_chars_nb
,
src
,
wctx
->
string_validation_replacement
);
}
end:
av_bprint_finalize
(
&
dstbuf
,
dstp
);
return
ret
;
}
#define PRINT_STRING_OPT 1
#define PRINT_STRING_VALIDATE 2
static
inline
int
writer_print_string
(
WriterContext
*
wctx
,
static
inline
int
writer_print_string
(
WriterContext
*
wctx
,
const
char
*
key
,
const
char
*
val
,
int
opt
)
const
char
*
key
,
const
char
*
val
,
int
flags
)
{
{
const
struct
section
*
section
=
wctx
->
section
[
wctx
->
level
];
const
struct
section
*
section
=
wctx
->
section
[
wctx
->
level
];
int
ret
=
0
;
int
ret
=
0
;
if
(
opt
&&
!
(
wctx
->
writer
->
flags
&
WRITER_FLAG_DISPLAY_OPTIONAL_FIELDS
))
if
((
flags
&
PRINT_STRING_OPT
)
&&
!
(
wctx
->
writer
->
flags
&
WRITER_FLAG_DISPLAY_OPTIONAL_FIELDS
))
return
0
;
return
0
;
if
(
section
->
show_all_entries
||
av_dict_get
(
section
->
entries_to_show
,
key
,
NULL
,
0
))
{
if
(
section
->
show_all_entries
||
av_dict_get
(
section
->
entries_to_show
,
key
,
NULL
,
0
))
{
if
(
flags
&
PRINT_STRING_VALIDATE
)
{
char
*
key1
=
NULL
,
*
val1
=
NULL
;
ret
=
validate_string
(
wctx
,
&
key1
,
key
);
if
(
ret
<
0
)
goto
end
;
ret
=
validate_string
(
wctx
,
&
val1
,
val
);
if
(
ret
<
0
)
goto
end
;
wctx
->
writer
->
print_string
(
wctx
,
key1
,
val1
);
end:
if
(
ret
<
0
)
{
av_log
(
wctx
,
AV_LOG_ERROR
,
"Invalid key=value string combination %s=%s in section %s
\n
"
,
key
,
val
,
section
->
unique_name
);
}
av_free
(
key1
);
av_free
(
val1
);
}
else
{
wctx
->
writer
->
print_string
(
wctx
,
key
,
val
);
wctx
->
writer
->
print_string
(
wctx
,
key
,
val
);
}
wctx
->
nb_item
[
wctx
->
level
]
++
;
wctx
->
nb_item
[
wctx
->
level
]
++
;
}
}
...
@@ -492,7 +623,7 @@ static void writer_print_time(WriterContext *wctx, const char *key,
...
@@ -492,7 +623,7 @@ static void writer_print_time(WriterContext *wctx, const char *key,
char
buf
[
128
];
char
buf
[
128
];
if
((
!
is_duration
&&
ts
==
AV_NOPTS_VALUE
)
||
(
is_duration
&&
ts
==
0
))
{
if
((
!
is_duration
&&
ts
==
AV_NOPTS_VALUE
)
||
(
is_duration
&&
ts
==
0
))
{
writer_print_string
(
wctx
,
key
,
"N/A"
,
1
);
writer_print_string
(
wctx
,
key
,
"N/A"
,
PRINT_STRING_OPT
);
}
else
{
}
else
{
double
d
=
ts
*
av_q2d
(
*
time_base
);
double
d
=
ts
*
av_q2d
(
*
time_base
);
struct
unit_value
uv
;
struct
unit_value
uv
;
...
@@ -506,7 +637,7 @@ static void writer_print_time(WriterContext *wctx, const char *key,
...
@@ -506,7 +637,7 @@ static void writer_print_time(WriterContext *wctx, const char *key,
static
void
writer_print_ts
(
WriterContext
*
wctx
,
const
char
*
key
,
int64_t
ts
,
int
is_duration
)
static
void
writer_print_ts
(
WriterContext
*
wctx
,
const
char
*
key
,
int64_t
ts
,
int
is_duration
)
{
{
if
((
!
is_duration
&&
ts
==
AV_NOPTS_VALUE
)
||
(
is_duration
&&
ts
==
0
))
{
if
((
!
is_duration
&&
ts
==
AV_NOPTS_VALUE
)
||
(
is_duration
&&
ts
==
0
))
{
writer_print_string
(
wctx
,
key
,
"N/A"
,
1
);
writer_print_string
(
wctx
,
key
,
"N/A"
,
PRINT_STRING_OPT
);
}
else
{
}
else
{
writer_print_integer
(
wctx
,
key
,
ts
);
writer_print_integer
(
wctx
,
key
,
ts
);
}
}
...
@@ -1476,7 +1607,8 @@ static void writer_register_all(void)
...
@@ -1476,7 +1607,8 @@ static void writer_register_all(void)
#define print_int(k, v) writer_print_integer(w, k, v)
#define print_int(k, v) writer_print_integer(w, k, v)
#define print_q(k, v, s) writer_print_rational(w, k, v, s)
#define print_q(k, v, s) writer_print_rational(w, k, v, s)
#define print_str(k, v) writer_print_string(w, k, v, 0)
#define print_str(k, v) writer_print_string(w, k, v, 0)
#define print_str_opt(k, v) writer_print_string(w, k, v, 1)
#define print_str_opt(k, v) writer_print_string(w, k, v, PRINT_STRING_OPT)
#define print_str_validate(k, v) writer_print_string(w, k, v, PRINT_STRING_VALIDATE)
#define print_time(k, v, tb) writer_print_time(w, k, v, tb, 0)
#define print_time(k, v, tb) writer_print_time(w, k, v, tb, 0)
#define print_ts(k, v) writer_print_ts(w, k, v, 0)
#define print_ts(k, v) writer_print_ts(w, k, v, 0)
#define print_duration_time(k, v, tb) writer_print_time(w, k, v, tb, 1)
#define print_duration_time(k, v, tb) writer_print_time(w, k, v, tb, 1)
...
@@ -1491,21 +1623,20 @@ static void writer_register_all(void)
...
@@ -1491,21 +1623,20 @@ static void writer_register_all(void)
#define print_section_header(s) writer_print_section_header(w, s)
#define print_section_header(s) writer_print_section_header(w, s)
#define print_section_footer(s) writer_print_section_footer(w, s)
#define print_section_footer(s) writer_print_section_footer(w, s)
static
inline
int
show_tags
(
WriterContext
*
w
ctx
,
AVDictionary
*
tags
,
int
section_id
)
static
inline
int
show_tags
(
WriterContext
*
w
,
AVDictionary
*
tags
,
int
section_id
)
{
{
AVDictionaryEntry
*
tag
=
NULL
;
AVDictionaryEntry
*
tag
=
NULL
;
int
ret
=
0
;
int
ret
=
0
;
if
(
!
tags
)
if
(
!
tags
)
return
0
;
return
0
;
writer_print_section_header
(
w
ctx
,
section_id
);
writer_print_section_header
(
w
,
section_id
);
while
((
tag
=
av_dict_get
(
tags
,
""
,
tag
,
AV_DICT_IGNORE_SUFFIX
)))
{
while
((
tag
=
av_dict_get
(
tags
,
""
,
tag
,
AV_DICT_IGNORE_SUFFIX
)))
{
ret
=
writer_print_string
(
wctx
,
tag
->
key
,
tag
->
value
,
0
);
if
((
ret
=
print_str_validate
(
tag
->
key
,
tag
->
value
))
<
0
)
if
(
ret
<
0
)
break
;
break
;
}
}
writer_print_section_footer
(
w
ctx
);
writer_print_section_footer
(
w
);
return
ret
;
return
ret
;
}
}
...
@@ -2054,7 +2185,7 @@ static int show_format(WriterContext *w, AVFormatContext *fmt_ctx)
...
@@ -2054,7 +2185,7 @@ static int show_format(WriterContext *w, AVFormatContext *fmt_ctx)
int
ret
=
0
;
int
ret
=
0
;
writer_print_section_header
(
w
,
SECTION_ID_FORMAT
);
writer_print_section_header
(
w
,
SECTION_ID_FORMAT
);
print_str
(
"filename"
,
fmt_ctx
->
filename
);
print_str
_validate
(
"filename"
,
fmt_ctx
->
filename
);
print_int
(
"nb_streams"
,
fmt_ctx
->
nb_streams
);
print_int
(
"nb_streams"
,
fmt_ctx
->
nb_streams
);
print_int
(
"nb_programs"
,
fmt_ctx
->
nb_programs
);
print_int
(
"nb_programs"
,
fmt_ctx
->
nb_programs
);
print_str
(
"format_name"
,
fmt_ctx
->
iformat
->
name
);
print_str
(
"format_name"
,
fmt_ctx
->
iformat
->
name
);
...
@@ -2755,6 +2886,9 @@ int main(int argc, char **argv)
...
@@ -2755,6 +2886,9 @@ int main(int argc, char **argv)
if
((
ret
=
writer_open
(
&
wctx
,
w
,
w_args
,
if
((
ret
=
writer_open
(
&
wctx
,
w
,
w_args
,
sections
,
FF_ARRAY_ELEMS
(
sections
)))
>=
0
)
{
sections
,
FF_ARRAY_ELEMS
(
sections
)))
>=
0
)
{
if
(
w
==
&
xml_writer
)
wctx
->
string_validation_utf8_flags
|=
AV_UTF8_FLAG_EXCLUDE_XML_INVALID_CONTROL_CODES
;
writer_print_section_header
(
wctx
,
SECTION_ID_ROOT
);
writer_print_section_header
(
wctx
,
SECTION_ID_ROOT
);
if
(
do_show_program_version
)
if
(
do_show_program_version
)
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment