[FFmpeg-devel] [PATCH 1/4] lavu/bprint: implement av_bprint_strftime().
Stefano Sabatini
stefasab at gmail.com
Thu Nov 15 21:02:09 CET 2012
On date Thursday 2012-11-15 17:13:55 +0100, Nicolas George encoded:
> TODO bump & APIchanges
>
> Signed-off-by: Nicolas George <nicolas.george at normalesup.org>
> ---
> libavutil/bprint.c | 55 +++++++++++++++++++++++++++++++++++++++++++++++++
> libavutil/bprint.h | 11 +++++++++-
> tests/ref/fate/bprint | 2 ++
> 3 files changed, 67 insertions(+), 1 deletion(-)
>
>
> Reworded the documentation, added comments.
>
>
> diff --git a/libavutil/bprint.c b/libavutil/bprint.c
> index 9d8e7c1..8710469 100644
> --- a/libavutil/bprint.c
> +++ b/libavutil/bprint.c
> @@ -21,6 +21,7 @@
> #include <stdarg.h>
> #include <stdio.h>
> #include <string.h>
> +#include <time.h>
> #include "avassert.h"
> #include "bprint.h"
> #include "common.h"
> @@ -129,6 +130,50 @@ void av_bprint_chars(AVBPrint *buf, char c, unsigned n)
> av_bprint_grow(buf, n);
> }
>
> +void av_bprint_strftime(AVBPrint *buf, const char *fmt, const struct tm *tm)
> +{
> + unsigned room;
> + size_t l;
> +
> + if (!*fmt)
> + return;
> + while (1) {
> + room = av_bprint_room(buf);
> + if (room && (l = strftime(buf->str + buf->len, room, fmt, tm)))
> + break;
> + /* strftime does not tell us how much room it would need: let us
> + retry with twice as much until the buffer is large enough */
> + room = !room ? strlen(fmt) + 1 :
> + room <= INT_MAX / 2 ? room * 2 : INT_MAX;
> + if (!room)
> + room = strlen(fmt) / 2 + 1;
Is this required? Shouldn't the previous assignment guarantee that
room is > 0?
> + if (av_bprint_alloc(buf, room)) {
> + /* impossible to grow, try to manage something useful anyway */
> + room = av_bprint_room(buf);
> + if (room < 1024) {
> + /* if strftime fails because the buffer has (almost) reached
> + its maximum size, let us try in a local buffer; 1k should
> + be enough to format any real date+time string */
> + char buf2[1024];
> + if ((l = strftime(buf2, sizeof(buf2), fmt, tm))) {
> + av_bprintf(buf, "%s", buf2);
> + return;
> + }
> + }
> + if (room) {
> + /* if anything else failed and the buffer is not already
> + truncated, let us add a stock string and force truncation */
> + static const char txt[] = "[truncated strftime output]";
> + memset(buf->str + buf->len, '!', room);
> + memcpy(buf->str + buf->len, txt, FFMIN(sizeof(txt) - 1, room));
> + av_bprint_grow(buf, room); /* force truncation */
> + }
> + return;
OK this or the "date is insanely long" solution.
> + }
> + }
> + av_bprint_grow(buf, l);
> +}
> +
> void av_bprint_get_buffer(AVBPrint *buf, unsigned size,
> unsigned char **mem, unsigned *actual_size)
> {
> @@ -201,6 +246,7 @@ int main(void)
> {
> AVBPrint b;
> char buf[256];
> + struct tm testtime = { .tm_year = 100, .tm_mon = 11, .tm_mday = 20 };
>
> av_bprint_init(&b, 0, -1);
> bprint_pascal(&b, 5);
> @@ -235,6 +281,15 @@ int main(void)
> bprint_pascal(&b, 25);
> printf("Long text count only buffer: %u/%u\n", (unsigned)strlen(buf), b.len);
>
> + av_bprint_init(&b, 0, -1);
> + av_bprint_strftime(&b, "%Y-%m-%d", &testtime);
> + printf("strftime full: %u/%u \"%s\"\n", (unsigned)strlen(buf), b.len, b.str);
> + av_bprint_finalize(&b, NULL);
> +
> + av_bprint_init(&b, 0, 8);
> + av_bprint_strftime(&b, "%Y-%m-%d", &testtime);
> + printf("strftime truncated: %u/%u \"%s\"\n", (unsigned)strlen(buf), b.len, b.str);
> +
> return 0;
> }
>
> diff --git a/libavutil/bprint.h b/libavutil/bprint.h
> index c09b61f..28e2dae 100644
> --- a/libavutil/bprint.h
> +++ b/libavutil/bprint.h
> @@ -116,7 +116,7 @@ void av_bprint_init(AVBPrint *buf, unsigned size_init, unsigned size_max);
> void av_bprint_init_for_buffer(AVBPrint *buf, char *buffer, unsigned size);
>
> /**
> - * Append a formated string to a print buffer.
> + * Append a formatted string to a print buffer.
> */
> void av_bprintf(AVBPrint *buf, const char *fmt, ...) av_printf_format(2, 3);
>
> @@ -125,6 +125,15 @@ void av_bprintf(AVBPrint *buf, const char *fmt, ...) av_printf_format(2, 3);
> */
> void av_bprint_chars(AVBPrint *buf, char c, unsigned n);
>
> +struct tm;
> +/**
> + * Append a formatted date and time to a print buffer.
> + * Note: due to poor design of the standard strftime function, it may
Nit: @note
> + * produce poor results if the format string expands to a very long text and
> + * the bprint buffer is near the limit stated by the size_max option.
missing documentation for fmt (which should mention the function
strftime()) and tm parameters
> + */
> +void av_bprint_strftime(AVBPrint *buf, const char *fmt, const struct tm *tm);
> +
> /**
> * Allocate bytes in the buffer for external use.
> *
> diff --git a/tests/ref/fate/bprint b/tests/ref/fate/bprint
> index e027fa1..b33c1ae 100644
> --- a/tests/ref/fate/bprint
> +++ b/tests/ref/fate/bprint
> @@ -12,3 +12,5 @@ Short text in automatic buffer: 174/174
> Long text in automatic buffer: 1000/2834
> Long text count only buffer: 0/2834
> Long text count only buffer: 255/2834
> +strftime full: 255/10 "2000-12-20"
> +strftime truncated: 255/10 "2000-12"
LGTM otherwise, thanks.
--
FFmpeg = Formidable and Fierce Maxi Philosofic Eretic Geisha
More information about the ffmpeg-devel
mailing list