mercredi 11 mai 2016
Do it at compile time !
I found that picture this morning in the CodeProject daily mail and I was thinking about the answer I will give ! And for me the best is let the compiler work as much as possible.... so :
vendredi 17 avril 2015
Bash script: extract info from ffmpeg output.
I already script that a several time and I think it's will be a good time to take a note and save a bit my mind for the next time !
When we use ffmpeg in some complex script in which we pipe decoded frame into another process i.e an encoder, it's really useful to get all input video and audio info. But from the ffmpeg -i output there is several regular expression that you can use:
- Extract widthxheigth:
- Extract frame-rate:
- Split RESINFO into Width and Heigth
HEIGHT=`echo $RESINFO | grep -oP '\K[0-9]+$'`
Feel free to comment and add request/proposal if you think that there is a better/other way to do it !
mercredi 25 mars 2015
My favorite 'interruption' ...
#if defined(WIN32) || defined(WIN64)
__asm int 3;
#endif
int is the asm instruction you use to generate a software interruption. Under windows the interruption 3 execution will trigger a breakpoint and from the outside you will have the usual popup saying:
click "debug" and select debugger you want use :)
It's a trick I often use when I want to also stop at start-up or at a fixed code point during a dev phase.
mercredi 25 février 2015
Some 'tricks' about ::ftruncate (for windows & linux)
Let's simplify the context.
We just want to open a file, write some text and close it. Later that file will be open again, we will read the text already present and run some text manipulation and rewrite the file.
As the file size and the amount of data we used were small, we use a real simple implementation that we can summarize with the following code.
#include
#include
#include
#include
#include
using namespace std;
int main() {
int fd = -1;
fd = ::open("sample.txt", O_CREAT | O_RDWR, S_IRWXG | S_IRWXO | S_IRWXU);
char* txt = "a small text to fill the file";
::write(fd, txt, strlen(txt));
::close(fd);
fd = ::open("sample.txt", O_CREAT | O_RDWR , S_IRWXG | S_IRWXO | S_IRWXU);
struct stat data;
bool ret = 0 == ::fstat(fd, &data);
if(ret && data.st_size>0)
{
char* buf = new char[data.st_size+1];
size_t bytesRead = ::read(fd, &buf[0], data.st_size);
delete [] buf;
}
::ftruncate(fd, 0);
char* txt2 = "another small text to fill the file";
*
::write(fd, txt2, strlen(txt2));
::close(fd);
return 0;
Note the call to 'ftruncate' function, we use it to remove the file content. I know that in a real simple code I could have done that in a several others ways (i.e. ::open with O_TRUNC) but consider the fact that our file is shared across multiple process and that the real implementation is a bit more complex.
But let's continue.... The point is that is didn't work as expected. In fact, if the file type was 'ASCII text' after the 1st write, after the second write the file mode changed to 'data' and it wrote the text 'in-binary'. If we don't perform the 'read' or the 'ftruncate' the file type was still the same 'ASCII text'. The solution came from the following: if we 'read' the whole file the reach the end and it may be possible that the file descriptor is in a strange state where it can find the file mode. So we just added a 'seek' before the call to 'ftruncate' and the issues was gone.
i.e. lseek(fd, 0, SEEK_SET);
Easy, but really weird and I didn't find any question and page mentioning that kind of issue ! but there is one :)
That post is related to the question I asked here
vendredi 23 janvier 2015
ffmpeg: all concatenation methods don't lead to a correct file
As I spend several hour comparing the different way to do that, I can now recommend to use one of those 2 simple methods if you have to concatenate files with the following properties:
- if they use the same container and codecs
- if they are consecutive in a way their PTS should not have discontinuities
copy /B 1.MTS+2.MTS out.MTS
or
It should be fast and it should not introduce weird PTS between 1 and 2 !ffmpeg
-
i
"concat:1.MTS|2.MTS"
-
an
-
vcodec copy out.ts
Observed with:
ffmpeg version N-68482-g92a596f Copyright (c) 2000-2014 the FFmpeg developers
built on Dec 16 2014 02:53:08 with gcc 4.9.2 (GCC)
configuration: --enable-gpl --enable-version3 --disable-w32threads --enable-avisynth --enable-bzlib --enable-fontconfig --enable-frei0r --enable-gnutls --enable-iconv --enable-libass --enable-libblu
ray --enable-libbs2b --enable-libcaca --enable-libfreetype --enable-libgme --enable-libgsm --enable-libilbc --enable-libmodplug --enable-libmp3lame --enable-libopencore-amrnb --enable-libopencore-amrw
b --enable-libopenjpeg --enable-libopus --enable-librtmp --enable-libschroedinger --enable-libsoxr --enable-libspeex --enable-libtheora --enable-libtwolame --enable-libvidstab --enable-libvo-aacenc --
enable-libvo-amrwbenc --enable-libvorbis --enable-libvpx --enable-libwavpack --enable-libwebp --enable-libx264 --enable-libx265 --enable-libxavs --enable-libxvid --enable-decklink --enable-zlib
libavutil 54. 15.100 / 54. 15.100
libavcodec 56. 15.100 / 56. 15.100
libavformat 56. 15.105 / 56. 15.105
libavdevice 56. 3.100 / 56. 3.100
libavfilter 5. 3.101 / 5. 3.101
libswscale 3. 1.101 / 3. 1.101
libswresample 1. 1.100 / 1. 1.100
libpostproc 53. 3.100 / 53. 3.100
CMAKE script retro-compatibility
I didn't post anything for a while, but in since few days, I found several topic for which I would like to write something (It's like a reminder for me ...)
I worked on product that we build on several different linux platform (Centos/Ubuntu/ etc ...). We use cmake and someone updated a script in a way it couldn't work anymore with older version of cmake (prior to 2.8.3). As I implemented a work-around, I wanted to share it because it was an annoying issue.
In cmake v2.8.3, a new "Useful variable" was introduced CMAKE_CURRENT_LIST_DIR.
If you had to use a cmake script in which that variable is used, cmake will not complain, it will just consider the variable as empty.
The work-around use the other function of cmake to initialize that value and let the rest of your script unchanged.
if("${CMAKE_CURRENT_LIST_DIR}" STREQUAL "")
message(STATUS "work-around to get the current folder for cmake version prior to 2.8.3")
get_filename_component(CMAKE_CURRENT_LIST_DIR "${CMAKE_CURRENT_LIST_FILE}" PATH)
endif()
It's quit simple, first I test if the value is empty, and if it's true I use get_filename_component cmake command to extract the current folder path from the CMAKE_CURRENT_LIST_FILE cmake variable.
lundi 5 janvier 2015
C++ STL copy and lvalue
A common mistake when using STL object and algorithm is to assume implementation details. For example when we use the ostringstream class, it may be easy to assume that the object contain a string member and that the str() function must return that data. If we just think like that and don’t look deeper at the str() function declaration, we can write the following code and suppose that the output will be correct.
std::ostringstream os; os << 1234 << std::endl;
std::copy(os.str().begin(), os.str().end(), std::back_inserter(outputContainer));
But it will not be correct and you cannot rely on the compiler to detect the issue because the syntax is correct. We pass 2 iterator on a string, but the issue is that they are not iterator on the ‘same’ string.
The function declaration of str() is:
string str() const;
and it means 2 things:
- the ostringstream object will not be modified by the call
- the returned string object is an lvalue (return by copy).
you can also read that post in which that trap is fully detailed….
But to summarize, every call to str() return a different string so the the begin and end iterator we provided in the previous piece of code aren’t related to the same string.
The only correct way to manage that situation is to create a copy of the string like below or to use your own ostringstream version like describe in that post.
std::ostringstream os; os << 1234 << std::endl;
std::string acpy = os.str();
std::copy(acpy.begin(), acpy.end(), std::back_inserter(outputContainer));
The conclusion is that when you use a STL object or an algorithm take attention at the declaration because if you don’t you may fall in some traps.