Hi,

Just some heads up about the porting of MLT to MSVC.
I built libmlt and libmlt++ successfully, but will have now to face the
modules. I manage not to bring changes to the code except some minor
ones (very little) but stuffed src/win32/. I've only tried with MSVC 15
(2017) so far.

Regards,
É

Le 18/01/2018 à 09:09, Élie Michel a écrit :
> Hi Dan,
> 
> Thanks for the feedback. Indeed, a compilation flag would be a better
> idea than adding #defines everywhere.
> 
> I totally agree with you on minimizing the changes of the source code
> itself. I try to stick to the build system, and the main exception is
> about adding files to src/win32 whenever the code uses posix related
> headers that msvc is missing. This has been by the way the what I spent
> most of the porting time on so far.
> 
> Since it seems that I'm heading to a correct direction, I'll keep on
> working on this and let you know when I get first results.
> 
> Regards,
> É
> 
> Le 18/01/2018 à 02:31, Dan Dennedy a écrit :
>> Hi,
>> This work looks like a good, serious effort. Some people have suggested
>> to switch to CMake, but it would be much work. I am intrigued to see
>> this in your patch - at least the start of it.
>>
>> Instead of adding #define HAVE_STRUCT_TIMESPEC to many files, what about
>> adding -DHAVE_STRUCT_TIMESPEC to CFLAGS on Windows? That is what I do in
>> the Shotcut build script.
>>
>> There has been a previous attempt or enquiry of sorts for MSVC also via
>> KDE for Windows, but at the time I rejected it as I did not want to get
>> stuck trying to maintain it. I think at the time there was much heavier
>> source changes required for MSVC's lack of support for C99. Now, that
>> has changed. I more open to this as long as there is not heavy source
>> code changes.
>>
>> Sorry, but I do not have any recommendations or further suggestions.
>>
>> On Tue, Jan 16, 2018 at 5:04 PM Élie Michel <e...@exppad.com
>> <mailto:e...@exppad.com>> wrote:
>>
>>     Hi,
>>
>>     I am currently trying to add a Kdenlive[1] package to the Craft[2]
>>     system for building KDE applications on Windows, and for this I would
>>     need to build MLT using, among others, MSVC.
>>
>>     I already started working on it; the current state of my patch is
>>     attached to this message. But since this turns out to be quite a piece
>>     of work, I'd like to know what you think about this, whether there have
>>     been previous attempts, and which would be your general recommandations
>>     for achieving this.
>>
>>     Thanks!
>>
>>     Best,
>>     Élie
>>
>>     [1]: https://kdenlive.org/
>>     [2]: https://community.kde.org/Craft
>>     
>> ------------------------------------------------------------------------------
>>     Check out the vibrant tech community on one of the world's most
>>     engaging tech sites, Slashdot.org!
>>     http://sdm.link/slashdot_______________________________________________
>>     Mlt-devel mailing list
>>     Mlt-devel@lists.sourceforge.net <mailto:Mlt-devel@lists.sourceforge.net>
>>     https://lists.sourceforge.net/lists/listinfo/mlt-devel
>>
>>
>>
>> ------------------------------------------------------------------------------
>> Check out the vibrant tech community on one of the world's most
>> engaging tech sites, Slashdot.org! http://sdm.link/slashdot
>>
>>
>>
>> _______________________________________________
>> Mlt-devel mailing list
>> Mlt-devel@lists.sourceforge.net
>> https://lists.sourceforge.net/lists/listinfo/mlt-devel
>>
> 
> ------------------------------------------------------------------------------
> Check out the vibrant tech community on one of the world's most
> engaging tech sites, Slashdot.org! http://sdm.link/slashdot
> _______________________________________________
> Mlt-devel mailing list
> Mlt-devel@lists.sourceforge.net
> https://lists.sourceforge.net/lists/listinfo/mlt-devel
> 
diff -Nru mlt-6.4.1-orig/cmake/modules/FindLibIconv.cmake 
mlt-6.4.1/cmake/modules/FindLibIconv.cmake
--- mlt-6.4.1-orig/cmake/modules/FindLibIconv.cmake     1970-01-01 
01:00:00.000000000 +0100
+++ mlt-6.4.1/cmake/modules/FindLibIconv.cmake  2018-01-22 09:24:14.832975500 
+0100
@@ -0,0 +1,15 @@
+# - Find the libIconv includes and library
+#
+# This module defines
+#  LIBICONV_INCLUDE_DIR, where to find iconv.h, etc.
+#  LIBICONV_LIBRARIES, the libraries to link against to use libIconv.
+#  LIBICONV_FOUND, If false, do not try to use libIconv.
+#=============================================================================
+#
+find_path(LIBICONV_INCLUDE_DIR iconv.h)
+find_library(LIBICONV_LIBRARIES NAMES iconv)
+# handle the QUIETLY and REQUIRED arguments and set LIBICONV_FOUND to TRUE if
+# all listed variables are TRUE
+include(FindPackageHandleStandardArgs)
+find_package_handle_standard_args(LibIconv  DEFAULT_MSG  LIBICONV_LIBRARIES 
LIBICONV_INCLUDE_DIR)
+mark_as_advanced(LIBICONV_INCLUDE_DIR LIBICONV_LIBRARIES)
diff -Nru mlt-6.4.1-orig/cmake/modules/FindPThreadsWin32.cmake 
mlt-6.4.1/cmake/modules/FindPThreadsWin32.cmake
--- mlt-6.4.1-orig/cmake/modules/FindPThreadsWin32.cmake        1970-01-01 
01:00:00.000000000 +0100
+++ mlt-6.4.1/cmake/modules/FindPThreadsWin32.cmake     2018-01-16 
21:14:40.640185900 +0100
@@ -0,0 +1,15 @@
+# - Find the native PThreadsWin32 includes and library
+#
+# This module defines
+#  PTHREADS_WIN32_INCLUDE_DIR, where to find pthread.h, etc.
+#  PTHREADS_WIN32_LIBRARIES, the libraries to link against to use 
PThreadsWin32.
+#  PTHREADS_WIN32_FOUND, If false, do not try to use PThreadsWin32.
+#=============================================================================
+#
+find_path(PTHREADS_WIN32_INCLUDE_DIR pthread.h)
+find_library(PTHREADS_WIN32_LIBRARIES NAMES pthreads)
+# handle the QUIETLY and REQUIRED arguments and set PTHREADS_WIN32_FOUND to 
TRUE if
+# all listed variables are TRUE
+include(FindPackageHandleStandardArgs)
+find_package_handle_standard_args(PThreadsWin32  DEFAULT_MSG  
PTHREADS_WIN32_LIBRARIES PTHREADS_WIN32_INCLUDE_DIR)
+mark_as_advanced(PTHREADS_WIN32_INCLUDE_DIR PTHREADS_WIN32_LIBRARIES)
diff -Nru mlt-6.4.1-orig/CMakeLists.txt mlt-6.4.1/CMakeLists.txt
--- mlt-6.4.1-orig/CMakeLists.txt       1970-01-01 01:00:00.000000000 +0100
+++ mlt-6.4.1/CMakeLists.txt    2018-01-22 23:21:38.765204900 +0100
@@ -0,0 +1,35 @@
+project(mlt)
+cmake_minimum_required(VERSION 3.0)
+
+set(CMAKE_MODULE_PATH ${CMAKE_SOURCE_DIR}/cmake/modules)
+
+include_directories(src)
+
+add_subdirectory(src/framework)
+add_subdirectory(src/mlt++)
+add_subdirectory(src/melt)
+#add_subdirectory(src/modules)
+#add_subdirectory(src/swig)
+#add_subdirectory(profiles)
+
+# Install
+
+#install:
+#      install -d "$(DESTDIR)$(prefix)/bin"
+#      install -d "$(DESTDIR)$(prefix)/include"
+#      install -d "$(DESTDIR)$(libdir)"
+#      install -d "$(DESTDIR)$(moduledir)"
+#ifeq ($(extra_versioning), true)
+#      ln -s "$(moduledir)" "$(DESTDIR)$(unversionedmoduledir)"
+#endif
+#      install -d "$(DESTDIR)$(libdir)/pkgconfig"
+#      install -d "$(DESTDIR)$(mltdatadir)"
+#ifeq ($(extra_versioning), true)
+#      ln -s "$(mltdatadir)" "$(DESTDIR)$(unversionedmltdatadir)"
+#endif
+#      install -c -m 644 *.pc "$(DESTDIR)$(libdir)/pkgconfig"
+#      list='$(SUBDIRS)'; \
+#      for subdir in $$list; do \
+#              $(MAKE) DESTDIR=$(DESTDIR) -C $$subdir $@ || exit 1; \
+#      done
+#      cp -R presets "$(DESTDIR)$(mltdatadir)"
diff -Nru mlt-6.4.1-orig/src/framework/CMakeLists.txt 
mlt-6.4.1/src/framework/CMakeLists.txt
--- mlt-6.4.1-orig/src/framework/CMakeLists.txt 1970-01-01 01:00:00.000000000 
+0100
+++ mlt-6.4.1/src/framework/CMakeLists.txt      2018-01-22 23:22:44.267978400 
+0100
@@ -0,0 +1,104 @@
+cmake_minimum_required(VERSION 3.0)
+
+find_package(Threads REQUIRED)
+if(CMAKE_USE_PTHREADS_INIT)
+       set(PTHREADS_WIN32_LIBRARIES ${CMAKE_THREAD_LIBS_INIT})
+       set(PTHREADS_WIN32_INCLUDE_DIR .)
+else(CMAKE_USE_PTHREADS_INIT)
+       find_package(PThreadsWin32 REQUIRED)
+endif(CMAKE_USE_PTHREADS_INIT)
+
+if(WIN32)
+       find_package(LibIconv REQUIRED)
+endif(WIN32)
+
+include_directories(${PTHREADS_WIN32_INCLUDE_DIR})
+
+# prevents error with timespec
+add_definitions(-DHAVE_STRUCT_TIMESPEC)
+
+set(SRC
+       mlt_frame.c
+       mlt_version.c
+       mlt_geometry.c
+       mlt_deque.c
+       mlt_property.c
+       mlt_properties.c
+       mlt_events.c
+       mlt_parser.c
+       mlt_service.c
+       mlt_producer.c
+       mlt_multitrack.c
+       mlt_playlist.c
+       mlt_consumer.c
+       mlt_filter.c
+       mlt_transition.c
+       mlt_field.c
+       mlt_tractor.c
+       mlt_factory.c
+       mlt_repository.c
+       mlt_pool.c
+       mlt_tokeniser.c
+       mlt_profile.c
+       mlt_log.c
+       mlt_cache.c
+       mlt_animation.c
+       mlt_slices.c
+       )
+
+set(INCS
+       mlt_consumer.h
+       mlt_version.h
+       mlt_factory.h
+       mlt_filter.h
+       mlt.h
+       mlt_multitrack.h
+       mlt_pool.h
+       mlt_properties.h
+       mlt_events.h
+       mlt_parser.h
+       mlt_repository.h
+       mlt_tractor.h
+       mlt_types.h
+       mlt_deque.h
+       mlt_field.h
+       mlt_frame.h
+       mlt_geometry.h
+       mlt_playlist.h
+       mlt_producer.h
+       mlt_property.h
+       mlt_service.h
+       mlt_transition.h
+       mlt_tokeniser.h
+       mlt_profile.h
+       mlt_log.h
+       mlt_cache.h
+       mlt_animation.h
+       mlt_slices.h
+       )
+
+if(WIN32)
+       include_directories(${CMAKE_CURRENT_SOURCE_DIR}/../win32)
+       set(SRC
+               ../win32/win32.c
+               ../win32/gettimeofday.c
+               ${SRC}
+       )
+endif(WIN32)
+
+if(BUILD_STATIC)
+       set(_SHARED STATIC)
+       add_definitions(-DREDLAND_STATIC)
+else(BUILD_STATIC)
+       set(_SHARED SHARED)
+endif(BUILD_STATIC)
+
+add_library(libmlt ${_SHARED} ${SRC})
+
+target_link_libraries(libmlt ${PTHREADS_WIN32_LIBRARIES})
+if(WIN32)
+       target_link_libraries(libmlt ${LIBICONV_LIBRARIES})
+endif(WIN32)
+
+
+#LDFLAGS += $(LIBDL) -lpthread -lm
diff -Nru mlt-6.4.1-orig/src/framework/mlt_consumer.c 
mlt-6.4.1/src/framework/mlt_consumer.c
--- mlt-6.4.1-orig/src/framework/mlt_consumer.c 2018-01-13 18:00:27.128248500 
+0100
+++ mlt-6.4.1/src/framework/mlt_consumer.c      2018-01-16 22:39:34.765741300 
+0100
@@ -30,7 +30,11 @@
 #include <stdio.h>
 #include <string.h>
 #include <stdlib.h>
+#ifdef _WIN32
+#include "../win32/gettimeofday.h"
+#else
 #include <sys/time.h>
+#endif /* ifdef _WIN32 */
 
 /** Define this if you want an automatic deinterlace (if necessary) when the
  * consumer's producer is not running at normal speed.
diff -Nru mlt-6.4.1-orig/src/framework/mlt_factory.c 
mlt-6.4.1/src/framework/mlt_factory.c
--- mlt-6.4.1-orig/src/framework/mlt_factory.c  2018-01-13 18:00:27.160128300 
+0100
+++ mlt-6.4.1/src/framework/mlt_factory.c       2018-01-16 23:45:25.016699400 
+0100
@@ -26,7 +26,7 @@
 #include <stdlib.h>
 #include <string.h>
 #include <locale.h>
-#include <libgen.h>
+//#include <libgen.h>
 
 /** the default subdirectory of the datadir for holding presets */
 #define PRESETS_DIR "/presets"
diff -Nru mlt-6.4.1-orig/src/framework/mlt_log.h 
mlt-6.4.1/src/framework/mlt_log.h
--- mlt-6.4.1-orig/src/framework/mlt_log.h      2018-01-13 18:00:27.219493400 
+0100
+++ mlt-6.4.1/src/framework/mlt_log.h   2018-01-16 21:32:32.665352100 +0100
@@ -77,13 +77,13 @@
 void mlt_log( void *service, int level, const char *fmt, ... );
 #endif
 
-#define mlt_log_panic(service, format, args...) mlt_log((service), 
MLT_LOG_PANIC, (format), ## args)
-#define mlt_log_fatal(service, format, args...) mlt_log((service), 
MLT_LOG_FATAL, (format), ## args)
-#define mlt_log_error(service, format, args...) mlt_log((service), 
MLT_LOG_ERROR, (format), ## args)
-#define mlt_log_warning(service, format, args...) mlt_log((service), 
MLT_LOG_WARNING, (format), ## args)
-#define mlt_log_info(service, format, args...) mlt_log((service), 
MLT_LOG_INFO, (format), ## args)
-#define mlt_log_verbose(service, format, args...) mlt_log((service), 
MLT_LOG_VERBOSE, (format), ## args)
-#define mlt_log_debug(service, format, args...) mlt_log((service), 
MLT_LOG_DEBUG, (format), ## args)
+#define mlt_log_panic(service, format, ...) mlt_log((service), MLT_LOG_PANIC, 
(format), ## __VA_ARGS__)
+#define mlt_log_fatal(service, format, ...) mlt_log((service), MLT_LOG_FATAL, 
(format), ## __VA_ARGS__)
+#define mlt_log_error(service, format, ...) mlt_log((service), MLT_LOG_ERROR, 
(format), ## __VA_ARGS__)
+#define mlt_log_warning(service, format, ...) mlt_log((service), 
MLT_LOG_WARNING, (format), ## __VA_ARGS__)
+#define mlt_log_info(service, format, ...) mlt_log((service), MLT_LOG_INFO, 
(format), ## __VA_ARGS__)
+#define mlt_log_verbose(service, format, ...) mlt_log((service), 
MLT_LOG_VERBOSE, (format), ## __VA_ARGS__)
+#define mlt_log_debug(service, format, ...) mlt_log((service), MLT_LOG_DEBUG, 
(format), ## __VA_ARGS__)
 
 void mlt_vlog( void *service, int level, const char *fmt, va_list );
 int mlt_log_get_level( void );
diff -Nru mlt-6.4.1-orig/src/framework/mlt_pool.c 
mlt-6.4.1/src/framework/mlt_pool.c
--- mlt-6.4.1-orig/src/framework/mlt_pool.c     2018-01-13 18:00:27.256112300 
+0100
+++ mlt-6.4.1/src/framework/mlt_pool.c  2018-01-22 08:52:10.984991800 +0100
@@ -68,7 +68,11 @@
  * optimized libraries (sse/altivec).
  */
 
+#if defined(_MSC_VER)
+typedef __declspec(align(16)) struct mlt_release_s
+#else
 typedef struct __attribute__ ((aligned (16))) mlt_release_s
+#endif
 {
        mlt_pool pool;
        int references;
diff -Nru mlt-6.4.1-orig/src/framework/mlt_profile.c 
mlt-6.4.1/src/framework/mlt_profile.c
--- mlt-6.4.1-orig/src/framework/mlt_profile.c  2018-01-13 18:00:27.277167800 
+0100
+++ mlt-6.4.1/src/framework/mlt_profile.c       2018-01-16 23:46:20.503960100 
+0100
@@ -24,7 +24,7 @@
 
 #include <stdlib.h>
 #include <string.h>
-#include <libgen.h>
+//#include <libgen.h>
 #include <math.h>
 
 
diff -Nru mlt-6.4.1-orig/src/framework/mlt_properties.c 
mlt-6.4.1/src/framework/mlt_properties.c
--- mlt-6.4.1-orig/src/framework/mlt_properties.c       2018-01-13 
18:00:27.287967900 +0100
+++ mlt-6.4.1/src/framework/mlt_properties.c    2018-01-22 08:51:13.917120200 
+0100
@@ -38,7 +38,7 @@
 #include <stdarg.h>
 #include <pthread.h>
 #include <sys/types.h>
-#include <dirent.h>
+#include "../win32/dirent.h"
 #include <sys/stat.h>
 #include <errno.h>
 #include <locale.h>
diff -Nru mlt-6.4.1-orig/src/framework/mlt_repository.c 
mlt-6.4.1/src/framework/mlt_repository.c
--- mlt-6.4.1-orig/src/framework/mlt_repository.c       2018-01-13 
18:00:27.308993700 +0100
+++ mlt-6.4.1/src/framework/mlt_repository.c    2018-01-22 23:15:37.598996000 
+0100
@@ -28,7 +28,7 @@
 
 #include <stdio.h>
 #include <stdlib.h>
-#include <dlfcn.h>
+#include "../win32/dlfcn.h"
 #include <string.h>
 #include <limits.h>
 #include <dirent.h>
@@ -102,11 +102,19 @@
                        flags |= RTLD_GLOBAL;
        
                // Open the shared object
-               void *object = dlopen( object_name, flags );
+#ifdef _WIN32
+               void *object = LoadLibrary( object_name );
+#else // _WIN32
+               HMODULE *object = dlopen( object_name, flags );
+#endif // _WIN32
                if ( object != NULL )
                {
                        // Get the registration function
+#ifdef _WIN32
+                       mlt_repository_callback symbol_ptr = GetProcAddress( 
object, "mlt_register" );
+#else // _WIN32
                        mlt_repository_callback symbol_ptr = dlsym( object, 
"mlt_register" );
+#endif // _WIN32
 
                        // Call the registration function
                        if ( symbol_ptr != NULL )
@@ -114,17 +122,29 @@
                                symbol_ptr( self );
 
                                // Register the object file for closure
+#ifdef _WIN32
+                       mlt_properties_set_data( &self->parent, object_name, 
object, 0, ( mlt_destructor )FreeLibrary, NULL );
+#else // _WIN32
                                mlt_properties_set_data( &self->parent, 
object_name, object, 0, ( mlt_destructor )dlclose, NULL );
+#endif // _WIN32
                                ++plugin_count;
                        }
                        else
                        {
+#ifdef _WIN32
+                               FreeLibrary( object );
+#else // _WIN32
                                dlclose( object );
+#endif // _WIN32
                        }
                }
                else if ( strstr( object_name, "libmlt" ) )
                {
-                       mlt_log_warning( NULL, "%s: failed to dlopen %s\n  
(%s)\n", __FUNCTION__, object_name, dlerror() );
+#ifdef _WIN32
+                               mlt_log_warning( NULL, "%s: failed to dlopen 
%s\n  (%s)\n", __FUNCTION__, object_name, "" );
+#else // _WIN32
+                               mlt_log_warning( NULL, "%s: failed to dlopen 
%s\n  (%s)\n", __FUNCTION__, object_name, dlerror() );
+#endif // _WIN32
                }
        }
 
diff -Nru mlt-6.4.1-orig/src/framework/mlt_slices.c 
mlt-6.4.1/src/framework/mlt_slices.c
--- mlt-6.4.1-orig/src/framework/mlt_slices.c   2018-01-13 18:00:27.330049200 
+0100
+++ mlt-6.4.1/src/framework/mlt_slices.c        2018-01-22 08:52:30.188042100 
+0100
@@ -23,7 +23,7 @@
 #include "mlt_slices.h"
 
 #include <stdlib.h>
-#include <unistd.h>
+//#include <unistd.h>
 #include <pthread.h>
 #ifdef _WIN32
 #include <windows.h>
diff -Nru mlt-6.4.1-orig/src/melt/CMakeLists.txt 
mlt-6.4.1/src/melt/CMakeLists.txt
--- mlt-6.4.1-orig/src/melt/CMakeLists.txt      1970-01-01 01:00:00.000000000 
+0100
+++ mlt-6.4.1/src/melt/CMakeLists.txt   2018-01-22 23:39:30.578057300 +0100
@@ -0,0 +1,37 @@
+cmake_minimum_required(VERSION 3.0)
+
+find_package(Threads REQUIRED)
+if(CMAKE_USE_PTHREADS_INIT)
+       set(PTHREADS_WIN32_LIBRARIES ${CMAKE_THREAD_LIBS_INIT})
+       set(PTHREADS_WIN32_INCLUDE_DIR .)
+else(CMAKE_USE_PTHREADS_INIT)
+       find_package(PThreadsWin32 REQUIRED)
+endif(CMAKE_USE_PTHREADS_INIT)
+
+include_directories(${PTHREADS_WIN32_INCLUDE_DIR})
+
+# prevents error with timespec
+add_definitions(-DHAVE_STRUCT_TIMESPEC)
+
+find_library(LIBMLT libmlt)
+
+find_package(SDL REQUIRED)
+include_directories(${SDL_INCLUDE_DIRS})
+
+set(SRC
+       melt.c
+       io.c
+)
+
+add_executable(melt ${SRC})
+
+target_link_libraries(melt {LIBMLT})
+target_link_libraries(melt ${PTHREADS_WIN32_LIBRARIES})
+target_link_libraries(melt ${SDL_LIBRARIES})
+
+
+IF(MINGW)
+#CFLAGS += $(shell  pkg-config --cflags sdl)
+#LDFLAGS += $(shell pkg-config --libs sdl)
+ENDIF(MINGW)
+
diff -Nru mlt-6.4.1-orig/src/melt/melt.c mlt-6.4.1/src/melt/melt.c
--- mlt-6.4.1-orig/src/melt/melt.c      2018-01-13 18:00:27.400323900 +0100
+++ mlt-6.4.1/src/melt/melt.c   2018-01-22 23:38:30.307654600 +0100
@@ -24,9 +24,11 @@
 #include <stdlib.h>
 #include <string.h>
 #include <sched.h>
-#include <libgen.h>
 #include <limits.h>
+#ifndef _WIN32
+#include <libgen.h>
 #include <unistd.h>
+#endif
 #include <signal.h>
 
 #include <framework/mlt.h>
diff -Nru mlt-6.4.1-orig/src/mlt++/CMakeLists.txt 
mlt-6.4.1/src/mlt++/CMakeLists.txt
--- mlt-6.4.1-orig/src/mlt++/CMakeLists.txt     1970-01-01 01:00:00.000000000 
+0100
+++ mlt-6.4.1/src/mlt++/CMakeLists.txt  2018-01-22 23:30:18.010775400 +0100
@@ -0,0 +1,48 @@
+cmake_minimum_required(VERSION 3.0)
+
+find_package(Threads REQUIRED)
+if(CMAKE_USE_PTHREADS_INIT)
+       set(PTHREADS_WIN32_LIBRARIES ${CMAKE_THREAD_LIBS_INIT})
+       set(PTHREADS_WIN32_INCLUDE_DIR .)
+else(CMAKE_USE_PTHREADS_INIT)
+       find_package(PThreadsWin32 REQUIRED)
+endif(CMAKE_USE_PTHREADS_INIT)
+
+include_directories(${PTHREADS_WIN32_INCLUDE_DIR})
+
+# prevents error with timespec
+add_definitions(-DHAVE_STRUCT_TIMESPEC)
+
+set(SRC
+       MltAnimation.cpp
+       MltConsumer.cpp
+       MltDeque.cpp
+       MltEvent.cpp
+       MltFactory.cpp
+       MltField.cpp
+       MltFilter.cpp
+       MltFilteredConsumer.cpp
+       MltFilteredProducer.cpp
+       MltFrame.cpp
+       MltGeometry.cpp
+       MltMultitrack.cpp
+       MltParser.cpp
+       MltPlaylist.cpp
+       MltProducer.cpp
+       MltProfile.cpp
+       MltProperties.cpp
+       MltPushConsumer.cpp
+       MltRepository.cpp
+       MltService.cpp
+       MltTokeniser.cpp
+       MltTractor.cpp
+       MltTransition.cpp
+       )
+
+add_library(libmlt++
+       ${SRC}
+)
+
+target_link_libraries(libmlt++ libmlt)
+target_link_libraries(libmlt++ ${PTHREADS_WIN32_LIBRARIES})
+
diff -Nru mlt-6.4.1-orig/src/win32/dirent.h mlt-6.4.1/src/win32/dirent.h
--- mlt-6.4.1-orig/src/win32/dirent.h   1970-01-01 01:00:00.000000000 +0100
+++ mlt-6.4.1/src/win32/dirent.h        2018-01-22 22:10:31.505666800 +0100
@@ -0,0 +1,1158 @@
+/*
+ * Dirent interface for Microsoft Visual Studio
+ *
+ * Copyright (C) 2006-2012 Toni Ronkko
+ * This file is part of dirent.  Dirent may be freely distributed
+ * under the MIT license.  For all details and documentation, see
+ * https://github.com/tronkko/dirent
+ */
+#ifndef DIRENT_H
+#define DIRENT_H
+
+/*
+ * Include windows.h without Windows Sockets 1.1 to prevent conflicts with
+ * Windows Sockets 2.0.
+ */
+#ifndef WIN32_LEAN_AND_MEAN
+#   define WIN32_LEAN_AND_MEAN
+#endif
+#include <windows.h>
+
+#include <stdio.h>
+#include <stdarg.h>
+#include <wchar.h>
+#include <string.h>
+#include <stdlib.h>
+#include <malloc.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <errno.h>
+
+/* Indicates that d_type field is available in dirent structure */
+#define _DIRENT_HAVE_D_TYPE
+
+/* Indicates that d_namlen field is available in dirent structure */
+#define _DIRENT_HAVE_D_NAMLEN
+
+/* Entries missing from MSVC 6.0 */
+#if !defined(FILE_ATTRIBUTE_DEVICE)
+#   define FILE_ATTRIBUTE_DEVICE 0x40
+#endif
+
+/* File type and permission flags for stat(), general mask */
+#if !defined(S_IFMT)
+#   define S_IFMT _S_IFMT
+#endif
+
+/* Directory bit */
+#if !defined(S_IFDIR)
+#   define S_IFDIR _S_IFDIR
+#endif
+
+/* Character device bit */
+#if !defined(S_IFCHR)
+#   define S_IFCHR _S_IFCHR
+#endif
+
+/* Pipe bit */
+#if !defined(S_IFFIFO)
+#   define S_IFFIFO _S_IFFIFO
+#endif
+
+/* Regular file bit */
+#if !defined(S_IFREG)
+#   define S_IFREG _S_IFREG
+#endif
+
+/* Read permission */
+#if !defined(S_IREAD)
+#   define S_IREAD _S_IREAD
+#endif
+
+/* Write permission */
+#if !defined(S_IWRITE)
+#   define S_IWRITE _S_IWRITE
+#endif
+
+/* Execute permission */
+#if !defined(S_IEXEC)
+#   define S_IEXEC _S_IEXEC
+#endif
+
+/* Pipe */
+#if !defined(S_IFIFO)
+#   define S_IFIFO _S_IFIFO
+#endif
+
+/* Block device */
+#if !defined(S_IFBLK)
+#   define S_IFBLK 0
+#endif
+
+/* Link */
+#if !defined(S_IFLNK)
+#   define S_IFLNK 0
+#endif
+
+/* Socket */
+#if !defined(S_IFSOCK)
+#   define S_IFSOCK 0
+#endif
+
+/* Read user permission */
+#if !defined(S_IRUSR)
+#   define S_IRUSR S_IREAD
+#endif
+
+/* Write user permission */
+#if !defined(S_IWUSR)
+#   define S_IWUSR S_IWRITE
+#endif
+
+/* Execute user permission */
+#if !defined(S_IXUSR)
+#   define S_IXUSR 0
+#endif
+
+/* Read group permission */
+#if !defined(S_IRGRP)
+#   define S_IRGRP 0
+#endif
+
+/* Write group permission */
+#if !defined(S_IWGRP)
+#   define S_IWGRP 0
+#endif
+
+/* Execute group permission */
+#if !defined(S_IXGRP)
+#   define S_IXGRP 0
+#endif
+
+/* Read others permission */
+#if !defined(S_IROTH)
+#   define S_IROTH 0
+#endif
+
+/* Write others permission */
+#if !defined(S_IWOTH)
+#   define S_IWOTH 0
+#endif
+
+/* Execute others permission */
+#if !defined(S_IXOTH)
+#   define S_IXOTH 0
+#endif
+
+/* Maximum length of file name */
+#if !defined(PATH_MAX)
+#   define PATH_MAX MAX_PATH
+#endif
+#if !defined(FILENAME_MAX)
+#   define FILENAME_MAX MAX_PATH
+#endif
+#if !defined(NAME_MAX)
+#   define NAME_MAX FILENAME_MAX
+#endif
+
+/* File type flags for d_type */
+#define DT_UNKNOWN 0
+#define DT_REG S_IFREG
+#define DT_DIR S_IFDIR
+#define DT_FIFO S_IFIFO
+#define DT_SOCK S_IFSOCK
+#define DT_CHR S_IFCHR
+#define DT_BLK S_IFBLK
+#define DT_LNK S_IFLNK
+
+/* Macros for converting between st_mode and d_type */
+#define IFTODT(mode) ((mode) & S_IFMT)
+#define DTTOIF(type) (type)
+
+/*
+ * File type macros.  Note that block devices, sockets and links cannot be
+ * distinguished on Windows and the macros S_ISBLK, S_ISSOCK and S_ISLNK are
+ * only defined for compatibility.  These macros should always return false
+ * on Windows.
+ */
+#if !defined(S_ISFIFO)
+#   define S_ISFIFO(mode) (((mode) & S_IFMT) == S_IFIFO)
+#endif
+#if !defined(S_ISDIR)
+#   define S_ISDIR(mode) (((mode) & S_IFMT) == S_IFDIR)
+#endif
+#if !defined(S_ISREG)
+#   define S_ISREG(mode) (((mode) & S_IFMT) == S_IFREG)
+#endif
+#if !defined(S_ISLNK)
+#   define S_ISLNK(mode) (((mode) & S_IFMT) == S_IFLNK)
+#endif
+#if !defined(S_ISSOCK)
+#   define S_ISSOCK(mode) (((mode) & S_IFMT) == S_IFSOCK)
+#endif
+#if !defined(S_ISCHR)
+#   define S_ISCHR(mode) (((mode) & S_IFMT) == S_IFCHR)
+#endif
+#if !defined(S_ISBLK)
+#   define S_ISBLK(mode) (((mode) & S_IFMT) == S_IFBLK)
+#endif
+
+/* Return the exact length of the file name without zero terminator */
+#define _D_EXACT_NAMLEN(p) ((p)->d_namlen)
+
+/* Return the maximum size of a file name */
+#define _D_ALLOC_NAMLEN(p) ((PATH_MAX)+1)
+
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+
+/* Wide-character version */
+struct _wdirent {
+    /* Always zero */
+    long d_ino;
+
+    /* File position within stream */
+    long d_off;
+
+    /* Structure size */
+    unsigned short d_reclen;
+
+    /* Length of name without \0 */
+    size_t d_namlen;
+
+    /* File type */
+    int d_type;
+
+    /* File name */
+    wchar_t d_name[PATH_MAX+1];
+};
+typedef struct _wdirent _wdirent;
+
+struct _WDIR {
+    /* Current directory entry */
+    struct _wdirent ent;
+
+    /* Private file data */
+    WIN32_FIND_DATAW data;
+
+    /* True if data is valid */
+    int cached;
+
+    /* Win32 search handle */
+    HANDLE handle;
+
+    /* Initial directory name */
+    wchar_t *patt;
+};
+typedef struct _WDIR _WDIR;
+
+/* Multi-byte character version */
+struct dirent {
+    /* Always zero */
+    long d_ino;
+
+    /* File position within stream */
+    long d_off;
+
+    /* Structure size */
+    unsigned short d_reclen;
+
+    /* Length of name without \0 */
+    size_t d_namlen;
+
+    /* File type */
+    int d_type;
+
+    /* File name */
+    char d_name[PATH_MAX+1];
+};
+typedef struct dirent dirent;
+
+struct DIR {
+    struct dirent ent;
+    struct _WDIR *wdirp;
+};
+typedef struct DIR DIR;
+
+
+/* Dirent functions */
+static DIR *opendir (const char *dirname);
+static _WDIR *_wopendir (const wchar_t *dirname);
+
+static struct dirent *readdir (DIR *dirp);
+static struct _wdirent *_wreaddir (_WDIR *dirp);
+
+static int readdir_r(
+    DIR *dirp, struct dirent *entry, struct dirent **result);
+static int _wreaddir_r(
+    _WDIR *dirp, struct _wdirent *entry, struct _wdirent **result);
+
+static int closedir (DIR *dirp);
+static int _wclosedir (_WDIR *dirp);
+
+static void rewinddir (DIR* dirp);
+static void _wrewinddir (_WDIR* dirp);
+
+static int scandir (const char *dirname, struct dirent ***namelist,
+    int (*filter)(const struct dirent*),
+    int (*compare)(const void *, const void *));
+
+static int alphasort (const struct dirent **a, const struct dirent **b);
+
+static int versionsort (const struct dirent **a, const struct dirent **b);
+
+
+/* For compatibility with Symbian */
+#define wdirent _wdirent
+#define WDIR _WDIR
+#define wopendir _wopendir
+#define wreaddir _wreaddir
+#define wclosedir _wclosedir
+#define wrewinddir _wrewinddir
+
+
+/* Internal utility functions */
+static WIN32_FIND_DATAW *dirent_first (_WDIR *dirp);
+static WIN32_FIND_DATAW *dirent_next (_WDIR *dirp);
+
+static int dirent_mbstowcs_s(
+    size_t *pReturnValue,
+    wchar_t *wcstr,
+    size_t sizeInWords,
+    const char *mbstr,
+    size_t count);
+
+static int dirent_wcstombs_s(
+    size_t *pReturnValue,
+    char *mbstr,
+    size_t sizeInBytes,
+    const wchar_t *wcstr,
+    size_t count);
+
+static void dirent_set_errno (int error);
+
+
+/*
+ * Open directory stream DIRNAME for read and return a pointer to the
+ * internal working area that is used to retrieve individual directory
+ * entries.
+ */
+static _WDIR*
+_wopendir(
+    const wchar_t *dirname)
+{
+    _WDIR *dirp = NULL;
+    int error;
+
+    /* Must have directory name */
+    if (dirname == NULL  ||  dirname[0] == '\0') {
+        dirent_set_errno (ENOENT);
+        return NULL;
+    }
+
+    /* Allocate new _WDIR structure */
+    dirp = (_WDIR*) malloc (sizeof (struct _WDIR));
+    if (dirp != NULL) {
+        DWORD n;
+
+        /* Reset _WDIR structure */
+        dirp->handle = INVALID_HANDLE_VALUE;
+        dirp->patt = NULL;
+        dirp->cached = 0;
+
+        /* Compute the length of full path plus zero terminator
+         *
+         * Note that on WinRT there's no way to convert relative paths
+         * into absolute paths, so just assume it is an absolute path.
+         */
+#       if defined(WINAPI_FAMILY) && (WINAPI_FAMILY == WINAPI_FAMILY_PHONE_APP)
+            n = wcslen(dirname);
+#       else
+            n = GetFullPathNameW (dirname, 0, NULL, NULL);
+#       endif
+
+        /* Allocate room for absolute directory name and search pattern */
+        dirp->patt = (wchar_t*) malloc (sizeof (wchar_t) * n + 16);
+        if (dirp->patt) {
+
+            /*
+             * Convert relative directory name to an absolute one.  This
+             * allows rewinddir() to function correctly even when current
+             * working directory is changed between opendir() and rewinddir().
+             *
+             * Note that on WinRT there's no way to convert relative paths
+             * into absolute paths, so just assume it is an absolute path.
+             */
+#           if defined(WINAPI_FAMILY) && (WINAPI_FAMILY == 
WINAPI_FAMILY_PHONE_APP)
+                wcsncpy_s(dirp->patt, n+1, dirname, n);
+#           else
+                n = GetFullPathNameW (dirname, n, dirp->patt, NULL);
+#           endif
+            if (n > 0) {
+                wchar_t *p;
+
+                /* Append search pattern \* to the directory name */
+                p = dirp->patt + n;
+                if (dirp->patt < p) {
+                    switch (p[-1]) {
+                    case '\\':
+                    case '/':
+                    case ':':
+                        /* Directory ends in path separator, e.g. c:\temp\ */
+                        /*NOP*/;
+                        break;
+
+                    default:
+                        /* Directory name doesn't end in path separator */
+                        *p++ = '\\';
+                    }
+                }
+                *p++ = '*';
+                *p = '\0';
+
+                /* Open directory stream and retrieve the first entry */
+                if (dirent_first (dirp)) {
+                    /* Directory stream opened successfully */
+                    error = 0;
+                } else {
+                    /* Cannot retrieve first entry */
+                    error = 1;
+                    dirent_set_errno (ENOENT);
+                }
+
+            } else {
+                /* Cannot retrieve full path name */
+                dirent_set_errno (ENOENT);
+                error = 1;
+            }
+
+        } else {
+            /* Cannot allocate memory for search pattern */
+            error = 1;
+        }
+
+    } else {
+        /* Cannot allocate _WDIR structure */
+        error = 1;
+    }
+
+    /* Clean up in case of error */
+    if (error  &&  dirp) {
+        _wclosedir (dirp);
+        dirp = NULL;
+    }
+
+    return dirp;
+}
+
+/*
+ * Read next directory entry.
+ *
+ * Returns pointer to static directory entry which may be overwritten by
+ * subsequent calls to _wreaddir().
+ */
+static struct _wdirent*
+_wreaddir(
+    _WDIR *dirp)
+{
+    struct _wdirent *entry;
+
+    /*
+     * Read directory entry to buffer.  We can safely ignore the return value
+     * as entry will be set to NULL in case of error.
+     */
+    (void) _wreaddir_r (dirp, &dirp->ent, &entry);
+
+    /* Return pointer to statically allocated directory entry */
+    return entry;
+}
+
+/*
+ * Read next directory entry.
+ *
+ * Returns zero on success.  If end of directory stream is reached, then sets
+ * result to NULL and returns zero.
+ */
+static int
+_wreaddir_r(
+    _WDIR *dirp,
+    struct _wdirent *entry,
+    struct _wdirent **result)
+{
+    WIN32_FIND_DATAW *datap;
+
+    /* Read next directory entry */
+    datap = dirent_next (dirp);
+    if (datap) {
+        size_t n;
+        DWORD attr;
+
+        /*
+         * Copy file name as wide-character string.  If the file name is too
+         * long to fit in to the destination buffer, then truncate file name
+         * to PATH_MAX characters and zero-terminate the buffer.
+         */
+        n = 0;
+        while (n < PATH_MAX  &&  datap->cFileName[n] != 0) {
+            entry->d_name[n] = datap->cFileName[n];
+            n++;
+        }
+        entry->d_name[n] = 0;
+
+        /* Length of file name excluding zero terminator */
+        entry->d_namlen = n;
+
+        /* File type */
+        attr = datap->dwFileAttributes;
+        if ((attr & FILE_ATTRIBUTE_DEVICE) != 0) {
+            entry->d_type = DT_CHR;
+        } else if ((attr & FILE_ATTRIBUTE_DIRECTORY) != 0) {
+            entry->d_type = DT_DIR;
+        } else {
+            entry->d_type = DT_REG;
+        }
+
+        /* Reset dummy fields */
+        entry->d_ino = 0;
+        entry->d_off = 0;
+        entry->d_reclen = sizeof (struct _wdirent);
+
+        /* Set result address */
+        *result = entry;
+
+    } else {
+
+        /* Return NULL to indicate end of directory */
+        *result = NULL;
+
+    }
+
+    return /*OK*/0;
+}
+
+/*
+ * Close directory stream opened by opendir() function.  This invalidates the
+ * DIR structure as well as any directory entry read previously by
+ * _wreaddir().
+ */
+static int
+_wclosedir(
+    _WDIR *dirp)
+{
+    int ok;
+    if (dirp) {
+
+        /* Release search handle */
+        if (dirp->handle != INVALID_HANDLE_VALUE) {
+            FindClose (dirp->handle);
+            dirp->handle = INVALID_HANDLE_VALUE;
+        }
+
+        /* Release search pattern */
+        if (dirp->patt) {
+            free (dirp->patt);
+            dirp->patt = NULL;
+        }
+
+        /* Release directory structure */
+        free (dirp);
+        ok = /*success*/0;
+
+    } else {
+
+        /* Invalid directory stream */
+        dirent_set_errno (EBADF);
+        ok = /*failure*/-1;
+
+    }
+    return ok;
+}
+
+/*
+ * Rewind directory stream such that _wreaddir() returns the very first
+ * file name again.
+ */
+static void
+_wrewinddir(
+    _WDIR* dirp)
+{
+    if (dirp) {
+        /* Release existing search handle */
+        if (dirp->handle != INVALID_HANDLE_VALUE) {
+            FindClose (dirp->handle);
+        }
+
+        /* Open new search handle */
+        dirent_first (dirp);
+    }
+}
+
+/* Get first directory entry (internal) */
+static WIN32_FIND_DATAW*
+dirent_first(
+    _WDIR *dirp)
+{
+    WIN32_FIND_DATAW *datap;
+
+    /* Open directory and retrieve the first entry */
+    dirp->handle = FindFirstFileExW(
+        dirp->patt, FindExInfoStandard, &dirp->data,
+        FindExSearchNameMatch, NULL, 0);
+    if (dirp->handle != INVALID_HANDLE_VALUE) {
+
+        /* a directory entry is now waiting in memory */
+        datap = &dirp->data;
+        dirp->cached = 1;
+
+    } else {
+
+        /* Failed to re-open directory: no directory entry in memory */
+        dirp->cached = 0;
+        datap = NULL;
+
+    }
+    return datap;
+}
+
+/*
+ * Get next directory entry (internal).
+ *
+ * Returns 
+ */
+static WIN32_FIND_DATAW*
+dirent_next(
+    _WDIR *dirp)
+{
+    WIN32_FIND_DATAW *p;
+
+    /* Get next directory entry */
+    if (dirp->cached != 0) {
+
+        /* A valid directory entry already in memory */
+        p = &dirp->data;
+        dirp->cached = 0;
+
+    } else if (dirp->handle != INVALID_HANDLE_VALUE) {
+
+        /* Get the next directory entry from stream */
+        if (FindNextFileW (dirp->handle, &dirp->data) != FALSE) {
+            /* Got a file */
+            p = &dirp->data;
+        } else {
+            /* The very last entry has been processed or an error occurred */
+            FindClose (dirp->handle);
+            dirp->handle = INVALID_HANDLE_VALUE;
+            p = NULL;
+        }
+
+    } else {
+
+        /* End of directory stream reached */
+        p = NULL;
+
+    }
+
+    return p;
+}
+
+/*
+ * Open directory stream using plain old C-string.
+ */
+static DIR*
+opendir(
+    const char *dirname) 
+{
+    struct DIR *dirp;
+    int error;
+
+    /* Must have directory name */
+    if (dirname == NULL  ||  dirname[0] == '\0') {
+        dirent_set_errno (ENOENT);
+        return NULL;
+    }
+
+    /* Allocate memory for DIR structure */
+    dirp = (DIR*) malloc (sizeof (struct DIR));
+    if (dirp) {
+        wchar_t wname[PATH_MAX + 1];
+        size_t n;
+
+        /* Convert directory name to wide-character string */
+        error = dirent_mbstowcs_s(
+            &n, wname, PATH_MAX + 1, dirname, PATH_MAX + 1);
+        if (!error) {
+
+            /* Open directory stream using wide-character name */
+            dirp->wdirp = _wopendir (wname);
+            if (dirp->wdirp) {
+                /* Directory stream opened */
+                error = 0;
+            } else {
+                /* Failed to open directory stream */
+                error = 1;
+            }
+
+        } else {
+            /*
+             * Cannot convert file name to wide-character string.  This
+             * occurs if the string contains invalid multi-byte sequences or
+             * the output buffer is too small to contain the resulting
+             * string.
+             */
+            error = 1;
+        }
+
+    } else {
+        /* Cannot allocate DIR structure */
+        error = 1;
+    }
+
+    /* Clean up in case of error */
+    if (error  &&  dirp) {
+        free (dirp);
+        dirp = NULL;
+    }
+
+    return dirp;
+}
+
+/*
+ * Read next directory entry.
+ */
+static struct dirent*
+readdir(
+    DIR *dirp)
+{
+    struct dirent *entry;
+
+    /*
+     * Read directory entry to buffer.  We can safely ignore the return value
+     * as entry will be set to NULL in case of error.
+     */
+    (void) readdir_r (dirp, &dirp->ent, &entry);
+
+    /* Return pointer to statically allocated directory entry */
+    return entry;
+}
+
+/*
+ * Read next directory entry into called-allocated buffer.
+ *
+ * Returns zero on success.  If the end of directory stream is reached, then
+ * sets result to NULL and returns zero.
+ */
+static int
+readdir_r(
+    DIR *dirp,
+    struct dirent *entry,
+    struct dirent **result)
+{
+    WIN32_FIND_DATAW *datap;
+
+    /* Read next directory entry */
+    datap = dirent_next (dirp->wdirp);
+    if (datap) {
+        size_t n;
+        int error;
+
+        /* Attempt to convert file name to multi-byte string */
+        error = dirent_wcstombs_s(
+            &n, entry->d_name, PATH_MAX + 1, datap->cFileName, PATH_MAX + 1);
+
+        /*
+         * If the file name cannot be represented by a multi-byte string,
+         * then attempt to use old 8+3 file name.  This allows traditional
+         * Unix-code to access some file names despite of unicode
+         * characters, although file names may seem unfamiliar to the user.
+         *
+         * Be ware that the code below cannot come up with a short file
+         * name unless the file system provides one.  At least
+         * VirtualBox shared folders fail to do this.
+         */
+        if (error  &&  datap->cAlternateFileName[0] != '\0') {
+            error = dirent_wcstombs_s(
+                &n, entry->d_name, PATH_MAX + 1,
+                datap->cAlternateFileName, PATH_MAX + 1);
+        }
+
+        if (!error) {
+            DWORD attr;
+
+            /* Length of file name excluding zero terminator */
+            entry->d_namlen = n - 1;
+
+            /* File attributes */
+            attr = datap->dwFileAttributes;
+            if ((attr & FILE_ATTRIBUTE_DEVICE) != 0) {
+                entry->d_type = DT_CHR;
+            } else if ((attr & FILE_ATTRIBUTE_DIRECTORY) != 0) {
+                entry->d_type = DT_DIR;
+            } else {
+                entry->d_type = DT_REG;
+            }
+
+            /* Reset dummy fields */
+            entry->d_ino = 0;
+            entry->d_off = 0;
+            entry->d_reclen = sizeof (struct dirent);
+
+        } else {
+
+            /*
+             * Cannot convert file name to multi-byte string so construct
+             * an erroneous directory entry and return that.  Note that
+             * we cannot return NULL as that would stop the processing
+             * of directory entries completely.
+             */
+            entry->d_name[0] = '?';
+            entry->d_name[1] = '\0';
+            entry->d_namlen = 1;
+            entry->d_type = DT_UNKNOWN;
+            entry->d_ino = 0;
+            entry->d_off = -1;
+            entry->d_reclen = 0;
+
+        }
+
+        /* Return pointer to directory entry */
+        *result = entry;
+
+    } else {
+
+        /* No more directory entries */
+        *result = NULL;
+
+    }
+
+    return /*OK*/0;
+}
+
+/*
+ * Close directory stream.
+ */
+static int
+closedir(
+    DIR *dirp)
+{
+    int ok;
+    if (dirp) {
+
+        /* Close wide-character directory stream */
+        ok = _wclosedir (dirp->wdirp);
+        dirp->wdirp = NULL;
+
+        /* Release multi-byte character version */
+        free (dirp);
+
+    } else {
+
+        /* Invalid directory stream */
+        dirent_set_errno (EBADF);
+        ok = /*failure*/-1;
+
+    }
+    return ok;
+}
+
+/*
+ * Rewind directory stream to beginning.
+ */
+static void
+rewinddir(
+    DIR* dirp)
+{
+    /* Rewind wide-character string directory stream */
+    _wrewinddir (dirp->wdirp);
+}
+
+/*
+ * Scan directory for entries.
+ */
+static int
+scandir(
+    const char *dirname,
+    struct dirent ***namelist,
+    int (*filter)(const struct dirent*),
+    int (*compare)(const void*, const void*))
+{
+    struct dirent **files = NULL;
+    size_t size = 0;
+    size_t allocated = 0;
+    const size_t init_size = 1;
+    DIR *dir = NULL;
+    struct dirent *entry;
+    struct dirent *tmp = NULL;
+    size_t i;
+    int result = 0;
+
+    /* Open directory stream */
+    dir = opendir (dirname);
+    if (dir) {
+
+        /* Read directory entries to memory */
+        while (1) {
+
+            /* Enlarge pointer table to make room for another pointer */
+            if (size >= allocated) {
+                void *p;
+                size_t num_entries;
+
+                /* Compute number of entries in the enlarged pointer table */
+                if (size < init_size) {
+                    /* Allocate initial pointer table */
+                    num_entries = init_size;
+                } else {
+                    /* Double the size */
+                    num_entries = size * 2;
+                }
+
+                /* Allocate first pointer table or enlarge existing table */
+                p = realloc (files, sizeof (void*) * num_entries);
+                if (p != NULL) {
+                    /* Got the memory */
+                    files = (dirent**) p;
+                    allocated = num_entries;
+                } else {
+                    /* Out of memory */
+                    result = -1;
+                    break;
+                }
+
+            }
+
+            /* Allocate room for temporary directory entry */
+            if (tmp == NULL) {
+                tmp = (struct dirent*) malloc (sizeof (struct dirent));
+                if (tmp == NULL) {
+                    /* Cannot allocate temporary directory entry */
+                    result = -1;
+                    break;
+                }
+            }
+
+            /* Read directory entry to temporary area */
+            if (readdir_r (dir, tmp, &entry) == /*OK*/0) {
+
+                /* Did we get an entry? */
+                if (entry != NULL) {
+                    int pass;
+
+                    /* Determine whether to include the entry in result */
+                    if (filter) {
+                        /* Let the filter function decide */
+                        pass = filter (tmp);
+                    } else {
+                        /* No filter function, include everything */
+                        pass = 1;
+                    }
+
+                    if (pass) {
+                        /* Store the temporary entry to pointer table */
+                        files[size++] = tmp;
+                        tmp = NULL;
+
+                        /* Keep up with the number of files */
+                        result++;
+                    }
+
+                } else {
+
+                    /*
+                     * End of directory stream reached => sort entries and
+                     * exit.
+                     */
+                    qsort (files, size, sizeof (void*), compare);
+                    break;
+
+                }
+
+            } else {
+                /* Error reading directory entry */
+                result = /*Error*/ -1;
+                break;
+            }
+
+        }
+
+    } else {
+        /* Cannot open directory */
+        result = /*Error*/ -1;
+    }
+
+    /* Release temporary directory entry */
+    if (tmp) {
+        free (tmp);
+    }
+
+    /* Release allocated memory on error */
+    if (result < 0) {
+        for (i = 0; i < size; i++) {
+            free (files[i]);
+        }
+        free (files);
+        files = NULL;
+    }
+
+    /* Close directory stream */
+    if (dir) {
+        closedir (dir);
+    }
+
+    /* Pass pointer table to caller */
+    if (namelist) {
+        *namelist = files;
+    }
+    return result;
+}
+
+/* Alphabetical sorting */
+static int
+alphasort(
+    const struct dirent **a, const struct dirent **b)
+{
+    return strcoll ((*a)->d_name, (*b)->d_name);
+}
+
+/* Sort versions */
+static int
+versionsort(
+    const struct dirent **a, const struct dirent **b)
+{
+    /* FIXME: implement strverscmp and use that */
+    return alphasort (a, b);
+}
+
+
+/* Convert multi-byte string to wide character string */
+static int
+dirent_mbstowcs_s(
+    size_t *pReturnValue,
+    wchar_t *wcstr,
+    size_t sizeInWords,
+    const char *mbstr,
+    size_t count)
+{
+    int error;
+
+#if defined(_MSC_VER)  &&  _MSC_VER >= 1400
+
+    /* Microsoft Visual Studio 2005 or later */
+    error = mbstowcs_s (pReturnValue, wcstr, sizeInWords, mbstr, count);
+
+#else
+
+    /* Older Visual Studio or non-Microsoft compiler */
+    size_t n;
+
+    /* Convert to wide-character string (or count characters) */
+    n = mbstowcs (wcstr, mbstr, sizeInWords);
+    if (!wcstr  ||  n < count) {
+
+        /* Zero-terminate output buffer */
+        if (wcstr  &&  sizeInWords) {
+            if (n >= sizeInWords) {
+                n = sizeInWords - 1;
+            }
+            wcstr[n] = 0;
+        }
+
+        /* Length of resulting multi-byte string WITH zero terminator */
+        if (pReturnValue) {
+            *pReturnValue = n + 1;
+        }
+
+        /* Success */
+        error = 0;
+
+    } else {
+
+        /* Could not convert string */
+        error = 1;
+
+    }
+
+#endif
+
+    return error;
+}
+
+/* Convert wide-character string to multi-byte string */
+static int
+dirent_wcstombs_s(
+    size_t *pReturnValue,
+    char *mbstr,
+    size_t sizeInBytes, /* max size of mbstr */
+    const wchar_t *wcstr,
+    size_t count)
+{
+    int error;
+
+#if defined(_MSC_VER)  &&  _MSC_VER >= 1400
+
+    /* Microsoft Visual Studio 2005 or later */
+    error = wcstombs_s (pReturnValue, mbstr, sizeInBytes, wcstr, count);
+
+#else
+
+    /* Older Visual Studio or non-Microsoft compiler */
+    size_t n;
+
+    /* Convert to multi-byte string (or count the number of bytes needed) */
+    n = wcstombs (mbstr, wcstr, sizeInBytes);
+    if (!mbstr  ||  n < count) {
+
+        /* Zero-terminate output buffer */
+        if (mbstr  &&  sizeInBytes) {
+            if (n >= sizeInBytes) {
+                n = sizeInBytes - 1;
+            }
+            mbstr[n] = '\0';
+        }
+
+        /* Length of resulting multi-bytes string WITH zero-terminator */
+        if (pReturnValue) {
+            *pReturnValue = n + 1;
+        }
+
+        /* Success */
+        error = 0;
+
+    } else {
+
+        /* Cannot convert string */
+        error = 1;
+
+    }
+
+#endif
+
+    return error;
+}
+
+/* Set errno variable */
+static void
+dirent_set_errno(
+    int error)
+{
+#if defined(_MSC_VER)  &&  _MSC_VER >= 1400
+
+    /* Microsoft Visual Studio 2005 and later */
+    _set_errno (error);
+
+#else
+
+    /* Non-Microsoft compiler or older Microsoft compiler */
+    errno = error;
+
+#endif
+}
+
+
+#ifdef __cplusplus
+}
+#endif
+#endif /*DIRENT_H*/
+
diff -Nru mlt-6.4.1-orig/src/win32/dlfcn.h mlt-6.4.1/src/win32/dlfcn.h
--- mlt-6.4.1-orig/src/win32/dlfcn.h    1970-01-01 01:00:00.000000000 +0100
+++ mlt-6.4.1/src/win32/dlfcn.h 2018-01-16 22:52:07.896430300 +0100
@@ -0,0 +1,59 @@
+/*
+ * dlfcn-win32
+ * Copyright (c) 2007 Ramiro Polla
+ *
+ * dlfcn-win32 is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * dlfcn-win32 is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with dlfcn-win32; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  
USA
+ */
+
+#ifndef DLFCN_H
+#define DLFCN_H
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#if defined(DLFCN_WIN32_EXPORTS)
+#   define DLFCN_EXPORT __declspec(dllexport)
+#else
+#   define DLFCN_EXPORT
+#endif
+
+/* POSIX says these are implementation-defined.
+ * To simplify use with Windows API, we treat them the same way.
+ */
+
+#define RTLD_LAZY   0
+#define RTLD_NOW    0
+
+#define RTLD_GLOBAL (1 << 1)
+#define RTLD_LOCAL  (1 << 2)
+
+/* These two were added in The Open Group Base Specifications Issue 6.
+ * Note: All other RTLD_* flags in any dlfcn.h are not standard compliant.
+ */
+
+#define RTLD_DEFAULT    0
+#define RTLD_NEXT       0
+
+DLFCN_EXPORT void *dlopen ( const char *file, int mode );
+DLFCN_EXPORT int   dlclose(void *handle);
+DLFCN_EXPORT void *dlsym(void *handle, const char *name);
+DLFCN_EXPORT char *dlerror(void);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* DLFCN_H */
diff -Nru mlt-6.4.1-orig/src/win32/gettimeofday.c 
mlt-6.4.1/src/win32/gettimeofday.c
--- mlt-6.4.1-orig/src/win32/gettimeofday.c     1970-01-01 01:00:00.000000000 
+0100
+++ mlt-6.4.1/src/win32/gettimeofday.c  2018-01-22 08:57:27.722870100 +0100
@@ -0,0 +1,46 @@
+/**
+ * \file mlt_consumer.c
+ * \brief abstraction for all consumer services
+ * \see mlt_consumer_s
+ *
+ * Copyright (C) 2003-2015 Meltytech, LLC
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  
USA
+ */
+
+#include "gettimeofday.h"
+
+int gettimeofday(struct timeval* p, void* tz) {
+    ULARGE_INTEGER ul; // As specified on MSDN.
+    FILETIME ft;
+
+    // Returns a 64-bit value representing the number of
+    // 100-nanosecond intervals since January 1, 1601 (UTC).
+    GetSystemTimeAsFileTime(&ft);
+
+    // Fill ULARGE_INTEGER low and high parts.
+    ul.LowPart = ft.dwLowDateTime;
+    ul.HighPart = ft.dwHighDateTime;
+    // Convert to microseconds.
+    ul.QuadPart /= 10ULL;
+    // Remove Windows to UNIX Epoch delta.
+    ul.QuadPart -= 11644473600000000ULL;
+    // Modulo to retrieve the microseconds.
+    p->tv_usec = (long) (ul.QuadPart % 1000000LL);
+    // Divide to retrieve the seconds.
+    p->tv_sec = (long) (ul.QuadPart / 1000000LL);
+
+    return 0;
+}
diff -Nru mlt-6.4.1-orig/src/win32/gettimeofday.h 
mlt-6.4.1/src/win32/gettimeofday.h
--- mlt-6.4.1-orig/src/win32/gettimeofday.h     1970-01-01 01:00:00.000000000 
+0100
+++ mlt-6.4.1/src/win32/gettimeofday.h  2018-01-22 08:57:32.848354300 +0100
@@ -0,0 +1,31 @@
+/**
+ * \file mlt_consumer.c
+ * \brief abstraction for all consumer services
+ * \see mlt_consumer_s
+ *
+ * Copyright (C) 2003-2015 Meltytech, LLC
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  
USA
+ */
+
+#ifndef _GETTIMEOFDAY_
+#define _GETTIMEOFDAY_
+
+#include <windows.h>
+#include <time.h>
+
+int gettimeofday(struct timeval* p, void* tz);
+
+#endif // _GETTIMEOFDAY_
diff -Nru mlt-6.4.1-orig/src/win32/win32.c mlt-6.4.1/src/win32/win32.c
--- mlt-6.4.1-orig/src/win32/win32.c    2018-01-13 18:00:30.588390300 +0100
+++ mlt-6.4.1/src/win32/win32.c 2018-01-22 23:07:09.074183000 +0100
@@ -165,3 +165,29 @@
        }
        return result;
 }
+
+char* dirname(char *path)
+{
+       static char path_buffer[_MAX_PATH];
+       char drive[_MAX_DRIVE];
+       char dir[_MAX_DIR];
+       char fname[_MAX_FNAME];
+       char ext[_MAX_EXT];
+
+       _splitpath(path, drive, dir, fname, ext );
+       _makepath(path_buffer, drive, dir, "", "");
+       return path_buffer;
+}
+
+char* basename(char *path)
+{
+       static char path_buffer[_MAX_PATH];
+       char drive[_MAX_DRIVE];
+       char dir[_MAX_DIR];
+       char fname[_MAX_FNAME];
+       char ext[_MAX_EXT];
+
+       _splitpath(path, drive, dir, fname, ext );
+       _makepath(path_buffer, "", "", fname, ext);
+       return path_buffer;
+}
------------------------------------------------------------------------------
Check out the vibrant tech community on one of the world's most
engaging tech sites, Slashdot.org! http://sdm.link/slashdot
_______________________________________________
Mlt-devel mailing list
Mlt-devel@lists.sourceforge.net
https://lists.sourceforge.net/lists/listinfo/mlt-devel

Reply via email to