[FFmpeg-devel] [PATCH 1/2] lavf/file: implement directory listing callbacks
Mariusz Szczepańczyk
mszczepanczyk at gmail.com
Mon Jun 22 00:01:32 CEST 2015
From: Lukasz Marek <lukasz.m.luki2 at gmail.com>
Signed-off-by: Lukasz Marek <lukasz.m.luki2 at gmail.com>
---
libavformat/file.c | 87 ++++++++++++++++++++++++++++++++++++++++++++++++++++++
1 file changed, 87 insertions(+)
diff --git a/libavformat/file.c b/libavformat/file.c
index 6511328..e617ac8 100644
--- a/libavformat/file.c
+++ b/libavformat/file.c
@@ -23,6 +23,7 @@
#include "libavutil/internal.h"
#include "libavutil/opt.h"
#include "avformat.h"
+#include <dirent.h>
#include <fcntl.h>
#if HAVE_IO_H
#include <io.h>
@@ -51,6 +52,7 @@ typedef struct FileContext {
int fd;
int trunc;
int blocksize;
+ DIR *dir;
} FileContext;
static const AVOption file_options[] = {
@@ -189,6 +191,88 @@ static int file_close(URLContext *h)
return close(c->fd);
}
+static int file_open_dir(URLContext *h)
+{
+ FileContext *c = h->priv_data;
+
+ c->dir = opendir(h->filename);
+ if (!c->dir)
+ return AVERROR(errno);
+
+ return 0;
+}
+
+static int file_read_dir(URLContext *h, AVIODirEntry **next)
+{
+ FileContext *c = h->priv_data;
+ struct dirent *dir;
+ char *fullpath = NULL;
+
+ *next = ff_alloc_dir_entry();
+ if (!*next)
+ return AVERROR(ENOMEM);
+ do {
+ errno = 0;
+ dir = readdir(c->dir);
+ if (!dir) {
+ av_freep(next);
+ return AVERROR(errno);
+ }
+ } while (!strcmp(dir->d_name, ".") || !strcmp(dir->d_name, ".."));
+
+ fullpath = av_append_path_component(h->filename, dir->d_name);
+ if (fullpath) {
+ struct stat st;
+ if (!stat(fullpath, &st)) {
+ (*next)->group_id = st.st_gid;
+ (*next)->user_id = st.st_uid;
+ (*next)->size = st.st_size;
+ (*next)->filemode = st.st_mode & 0777;
+ (*next)->modification_timestamp = INT64_C(1000000) * st.st_mtime;
+ (*next)->access_timestamp = INT64_C(1000000) * st.st_atime;
+ (*next)->status_change_timestamp = INT64_C(1000000) * st.st_ctime;
+ }
+ av_free(fullpath);
+ }
+
+ (*next)->name = av_strdup(dir->d_name);
+ switch (dir->d_type) {
+ case DT_FIFO:
+ (*next)->type = AVIO_ENTRY_NAMED_PIPE;
+ break;
+ case DT_CHR:
+ (*next)->type = AVIO_ENTRY_CHARACTER_DEVICE;
+ break;
+ case DT_DIR:
+ (*next)->type = AVIO_ENTRY_DIRECTORY;
+ break;
+ case DT_BLK:
+ (*next)->type = AVIO_ENTRY_BLOCK_DEVICE;
+ break;
+ case DT_REG:
+ (*next)->type = AVIO_ENTRY_FILE;
+ break;
+ case DT_LNK:
+ (*next)->type = AVIO_ENTRY_SYMBOLIC_LINK;
+ break;
+ case DT_SOCK:
+ (*next)->type = AVIO_ENTRY_SOCKET;
+ break;
+ case DT_UNKNOWN:
+ default:
+ (*next)->type = AVIO_ENTRY_UNKNOWN;
+ break;
+ }
+ return 0;
+}
+
+static int file_close_dir(URLContext *h)
+{
+ FileContext *c = h->priv_data;
+ closedir(c->dir);
+ return 0;
+}
+
URLProtocol ff_file_protocol = {
.name = "file",
.url_open = file_open,
@@ -200,6 +284,9 @@ URLProtocol ff_file_protocol = {
.url_check = file_check,
.priv_data_size = sizeof(FileContext),
.priv_data_class = &file_class,
+ .url_open_dir = file_open_dir,
+ .url_read_dir = file_read_dir,
+ .url_close_dir = file_close_dir,
};
#endif /* CONFIG_FILE_PROTOCOL */
--
2.3.6
More information about the ffmpeg-devel
mailing list