ShaunR Posted September 8, 2021 Report Posted September 8, 2021 (edited) 15 minutes ago, hooovahh said: Very neat, glad you found an improvement. But it seems like you forgot to attach the code. Are you still making improvements? Sorry. No can do. It's part of ECL but I've given you all the info to replicate it and proved it might be worth your while Edited September 8, 2021 by ShaunR Quote
hooovahh Posted September 8, 2021 Author Report Posted September 8, 2021 26 minutes ago, ShaunR said: Sorry. No can do. It's part of ECL but I've given you all the info to replicate it and proved it might be worth your while I'm pretty satisfied with the solutions posted. You don't seem to be based on your responses. Quote
ShaunR Posted September 8, 2021 Report Posted September 8, 2021 24 minutes ago, hooovahh said: I'm pretty satisfied with the solutions posted. You don't seem to be based on your responses. I have a different solution... 1 hour ago, hooovahh said: Well this code has been here for 12 years ready to be improved, .. which is demonstrably an improvement. You can lead a horse to water but, this time, I guess I underestimated it's thirstiness. Quote
mahgust Posted September 8, 2021 Report Posted September 8, 2021 (edited) ShaunR, thanks for the idea sharing! Hoovah, code is attached (was deleted because of memory error with large >1GB files, use new version (MD5_my5.vi) posted in message below). Edited September 9, 2021 by mahgust 1 Quote
mahgust Posted September 8, 2021 Report Posted September 8, 2021 3 hours ago, ShaunR said: Yuck. Cmd line Try using the EVP_Digest interface of the NIlibeay32.dll You can even have progress events if you want to be fancy I don't know why NI didn't use it Direct MD5 call was used from NIlibeay32.dll instead of EVP_Digest interface Quote
hooovahh Posted September 8, 2021 Author Report Posted September 8, 2021 1 hour ago, ShaunR said: You can lead a horse to water but You most certainly can. I asked for you to share your improvement so everyone can benefit, you just seemed interested in talking about it but not showing it. 32 minutes ago, mahgust said: ShaunR, thanks for the idea sharing! Hoovah, code is attached. MD5_my2.vi 10.22 kB · 0 downloads Unfortunately I get "Not Enough Memory to complete this operation" error when I try it on my 1GB+ file. I suspect the other functions in that call library node are needed like Init, Update, and Final. Quote
mahgust Posted September 8, 2021 Report Posted September 8, 2021 10 minutes ago, hooovahh said: Unfortunately I get "Not Enough Memory to complete this operation" error when I try it on my 1GB+ file. I suspect the other functions in that call library node are needed like Init, Update, and Final. Yes, it seems like for a large files it should be done in this way http://websites.umich.edu/~x509/ssleay/md5.html Quote
ShaunR Posted September 8, 2021 Report Posted September 8, 2021 (edited) 1 hour ago, hooovahh said: You most certainly can. I asked for you to share your improvement so everyone can benefit, you just seemed interested in talking about it but not showing it. As I explained. The code I used is part of ECL so commercial IP prevents me from sharing. You should know or at least appreciate this! 1 hour ago, hooovahh said: I suspect the other functions in that call library node are needed like Init, Update, and Final. Correct. New, Init, update, final and free (the EVP interface). You will also need digestbyname. Once you have this you will also be able to do *all* the hashes, not just MD5. Edited September 8, 2021 by ShaunR Quote
hooovahh Posted September 8, 2021 Author Report Posted September 8, 2021 11 minutes ago, ShaunR said: As I explained. The code I used is part of ECL so commercial IP prevents me from sharing. Sorry if it wasn't clear from the original post, or my replies, but a commercial solution isn't appropriate for this. Quote
ShaunR Posted September 8, 2021 Report Posted September 8, 2021 (edited) 1 hour ago, mahgust said: Yes, it seems like for a large files it should be done in this way http://websites.umich.edu/~x509/ssleay/md5.html That is the old way. The EVP_Digest interface abstracts away the different hashes into unified set of functions. All that is needed is a while loop where they have two update functions in the example about 2/3rds of the way down in that link.. Edited September 8, 2021 by ShaunR Quote
ShaunR Posted September 8, 2021 Report Posted September 8, 2021 (edited) 6 minutes ago, hooovahh said: Sorry if it wasn't clear from the original post, or my replies, but a commercial solution isn't appropriate for this. I don't think you can see the wood for the trees. I am saying I can't supply the actual code I used because it's commercial but I can help you "discover" how to do it which isn't bound by commercial restraints. Edited September 8, 2021 by ShaunR Quote
mahgust Posted September 9, 2021 Report Posted September 9, 2021 New version with sequence Init -> Update -> Final. P.S. I tried EVP interface but is in stuck with EVP_MD_CTX structure definition in LV. https://github.com/theos/headers/blob/master/openssl/evp.h MD5_my5.vi 2 Quote
Rolf Kalbermatter Posted September 9, 2021 Report Posted September 9, 2021 (edited) 4 hours ago, mahgust said: New version with sequence Init -> Update -> Final. P.S. I tried EVP interface but is in stuck with EVP_MD_CTX structure definition in LV. https://github.com/theos/headers/blob/master/openssl/evp.h Since you don't access the internal elements in the struct at all from LabVIEW you just can treat it all as a pointer sized integer. In fact since OpenSSL 1.0.0 all those structs are considered opaque in terms of external users of the API and should never be referenced in any way other than through published OpenSSL functions. In terms of an external API user these contexts are meant to be simply a handle (a pointer to private data whose contents is unknown). EVP_MD_CTX_create() creates the context -> just configure it to return a pointer sized integer. Then pass this to all other EVP functions again as pointer sized integer. And of course don't forget to call the EVP_MD_CTX_free() function at the end to avoid memory leaks. Edited September 9, 2021 by Rolf Kalbermatter 1 Quote
mahgust Posted September 9, 2021 Report Posted September 9, 2021 2 hours ago, Rolf Kalbermatter said: Since you don't access the internal elements in the struct at all from LabVIEW you just can treat it all as a pointer sized integer. In fact since OpenSSL 1.0.0 all those structs are considered opaque in terms of external users of the API and should never be referenced in any way other than through published OpenSSL functions. In terms of an external API user these contexts are meant to be simply a handle (a pointer to private data whose contents is unknown). EVP_MD_CTX_create() creates the context -> just configure it to return a pointer sized integer. Then pass this to all other EVP functions again as pointer sized integer. And of course don't forget to call the EVP_MD_CTX_free() function at the end to avoid memory leaks. //new 3.0 openssl lib has new-free definitions mdctx = EVP_MD_CTX_new(); EVP_MD_CTX_free(mdctx); //old 1.0 openssl lib has create-destroy definitions mdctx = EVP_MD_CTX_create(); EVP_MD_CTX_destroy(mdctx); Lib in LV2014 operates in create/destroy definitions. I've tried to implement create-destroy without structure definition only with pointer, but it leads to LV crash. Could you suggest what is wrong with my code ? Thx! Untitled 1.vi Quote
Rolf Kalbermatter Posted September 9, 2021 Report Posted September 9, 2021 49 minutes ago, mahgust said: //new 3.0 openssl lib has new-free definitions mdctx = EVP_MD_CTX_new(); EVP_MD_CTX_free(mdctx); //old 1.0 openssl lib has create-destroy definitions mdctx = EVP_MD_CTX_create(); EVP_MD_CTX_destroy(mdctx); Lib in LV2014 operates in create/destroy definitions. I've tried to implement create-destroy without structure definition only with pointer, but it leads to LV crash. Could you suggest what is wrong with my code ? Thx! Untitled 1.vi 6.97 kB · 2 downloads One obvious discrepancy: create uses a pointer sized integer and destroy uses an Adapt to type. This will result in passing the pointer as an u64 passed by reference (Adapt to Type are always passed by reference if they are not handles, arrays or ActiveX references). What you want to configure it to is Numeric, Pointer sized Integer, Pass by Value. Yes you want to pass it by value, the value returned from the create function is already a pointer and destroy expects this pointer. Quote
ShaunR Posted September 9, 2021 Report Posted September 9, 2021 53 minutes ago, mahgust said: //new 3.0 openssl lib has new-free definitions mdctx = EVP_MD_CTX_new(); EVP_MD_CTX_free(mdctx); //old 1.0 openssl lib has create-destroy definitions mdctx = EVP_MD_CTX_create(); EVP_MD_CTX_destroy(mdctx); Lib in LV2014 operates in create/destroy definitions. I've tried to implement create-destroy without structure definition only with pointer, but it leads to LV crash. Could you suggest what is wrong with my code ? Thx! Untitled 1.vi 6.97 kB · 0 downloads The destroy should be an Unsigned Pointer Sized integer (Passed by value) rather than adapt to type. You can also set the nodes to "Run in any thread" Rolf got there 10 secs before me.lol Untitled 1.vi Quote
mahgust Posted September 9, 2021 Report Posted September 9, 2021 Thanks a lot, guys! EVP is working now! md4 md5 sha1 sha224 sha256 sha384 sha512 can be calculated. Quite convinient and fast utility for file integrity check can be created in this way. Should we point NI to this (hash calc through EVP interface) via Idea Exchange ? MD5_my7.vi 1 Quote
mahgust Posted September 9, 2021 Report Posted September 9, 2021 EVP Functions was used according this OpenSSL version https://www.openssl.org/docs/man1.0.2/man3/EVP_DigestInit.html OpenSSL_add_all_digests(); md = EVP_get_digestbyname(argv[1]); mdctx = EVP_MD_CTX_create(); EVP_DigestInit_ex(mdctx, md, NULL); EVP_DigestUpdate(mdctx, mess1, strlen(mess1)); EVP_DigestFinal_ex(mdctx, md_value, &md_len); EVP_MD_CTX_destroy(mdctx); EVP_cleanup(); Quote
hooovahh Posted September 9, 2021 Author Report Posted September 9, 2021 1 hour ago, mahgust said: Thanks a lot, guys! EVP is working now! md4 md5 sha1 sha224 sha256 sha384 sha512 can be calculated. Quite convinient and fast utility for file integrity check can be created in this way. Sweet. My test file was about 4.5s via command line and was 2.2s using your code. I'm pretty sure this is just copied over from the NI example, but we can get rid of all the Get and Set file positions. Not sure why NI put them there in the first place. Also is this NIlibeay binary available on other RT targets? I looked at the SSL functions used by Webdav, and the HTTP functions, and they call other NI binaries, the ni_webdavLVClient and ni_httpClient files that likely have a .so file or .dll based on target selection. I see Pharlap versions of this library but not any others. Quote
ShaunR Posted September 9, 2021 Report Posted September 9, 2021 2 hours ago, mahgust said: Thanks a lot, guys! EVP is working now! md4 md5 sha1 sha224 sha256 sha384 sha512 can be calculated. Quite convinient and fast utility for file integrity check can be created in this way. Should we point NI to this (hash calc through EVP interface) via Idea Exchange ? MD5_my7.vi 26.48 kB · 2 downloads Nice work. Glad you got there. Quote
ShaunR Posted September 9, 2021 Report Posted September 9, 2021 49 minutes ago, hooovahh said: Also is this NIlibeay binary available on other RT targets? There is an opk package in one of the feeds if it's not there already. Chances are, if it has SSH, it'll have compatible binaries. NIlibeay is just NI's compilation of OpenSSL. Quote
Lipko Posted September 10, 2021 Report Posted September 10, 2021 I looked at the code and one question (with low relevancy to the thread): Isn't get file size slow? I remember it being almost as slow as to read the whole file. Or did I do something wrong? Would listening to End Of File error be a better option or it has some caveats? Quote
Rolf Kalbermatter Posted September 10, 2021 Report Posted September 10, 2021 (edited) 11 hours ago, mahgust said: Thanks a lot, guys! EVP is working now! md4 md5 sha1 sha224 sha256 sha384 sha512 can be calculated. Quite convinient and fast utility for file integrity check can be created in this way. Should we point NI to this (hash calc through EVP interface) via Idea Exchange ? MD5_my7.viFetching info... You can remove the Set and Get File Offset inside the loop. The LabVIEW file IO nodes maintain internally a file offset (actually it's the underlying OS file IO functions which do and advance that pointer along as you read). As long as you do pure sequential access there is no need to set the file offset explicitly setting. It's even so that when you open a file for anything but append mode, the file offset will be automatically set to 0. Only when you do random access will you need to do explicit file offset setting. I don't expect this to save a lot of time but why do it if it is not necessary? Quote I looked at the code and one question (with low relevancy to the thread): Isn't get file size slow? I remember it being almost as slow as to read the whole file. Or did I do something wrong? Would listening to End Of File error be a better option or it has some caveats? That would seem very strange. The Get File Size directly translates to a Windows API call on the underlying file handle. Why that would be so slow is a miracle to me. Edited September 10, 2021 by Rolf Kalbermatter Quote
Rolf Kalbermatter Posted September 10, 2021 Report Posted September 10, 2021 2 hours ago, Lipko said: 9 hours ago, hooovahh said: I'm pretty sure this is just copied over from the NI example, but we can get rid of all the Get and Set file positions. Not sure why NI put them there in the first place. Most likely because the original code originates from pre LabVIEW 8.0. There all LabVIEW Read and Write nodes had explicit file offset input and output. When you upgrade these VIs, LabVIEW mutates them by adding explicit file offset calls before and after the File Read and File Write. It's the only safe way as LabVIEW can't easily know if the original file offset handling was unnecessary because the access is fully sequential or not. Obviously for trivial cases like this the analyzer could be made smart enough to decide that it is not needed, but there are corner cases where this is not easily decided. Rather than try to think up of all such corner cases and make sure that analyzer won't decide wrong by removing one file offset call to much, the easier thing is to simply maintain the original functionality and risk some performance loss (which is minimal in comparison to the old situation where this offset handling was always done anyways). The "example scrubber" for that code probably cleaned it up but didn't dare to remove the file offset calls, obviously not to familiar with LabVIEW internas. Quote
Lipko Posted September 10, 2021 Report Posted September 10, 2021 38 minutes ago, Rolf Kalbermatter said: That would seem very strange. The Get File Size directly translates to a Windows API call on the underlying file handle. Why that would be so slow is a miracle to me. Never mind, maybe I used File/directory info.vi. Quote
Recommended Posts
Join the conversation
You can post now and register later. If you have an account, sign in now to post with your account.