Monday, June 30, 2014

Memory management in C and auto allocating sprintf() - asprintf()

Memory Management

Memory management in C is viewed by some to be quite tricky. One needs to work with pointers that can point anywhere in memory, and if misused, cause a program to misbehave, or worse.

The basic functions to allocate and deallocate memory in C are malloc() and free() respectively. The malloc() function takes a size in bytes of how much to allocate, and returns a pointer to the allocated memory to be used. Upon failure, a null pointer, which can be thought of as a pointer pointing to 0 is returned. The free() function takes the pointer returned by malloc(), and deallocates the memory, or frees up the memory it was once pointing to.

To work with malloc() in simple situations, typically, code along the following lines is used:
void *p = malloc(size);
if (p)
{
  ... work with p ...
  free(p);
}
else
{
  ... handle error scenario ...
}

Unfortunately many experienced programmers forget to handle the failure scenario. I've even heard some say they purposely don't, as they have no clue how to proceed, and just letting the program crash is good enough for them. If you meet someone who makes that argument, revoke their programming license. We don't need such near sighted idiots writing libraries.

In any case, even the above can lead to some misuse. After this block of code runs, what is p now pointing to?

After the above code runs, in the case that malloc() succeeded, p is now pointing to memory in middle of nowhere, and can't be used. This is known as a dangling pointer. Dangling pointers can be dangerous, as an if clause as above will think the pointer is valid, and may do something with it, or lead to the infamous use after free bug. This becomes more likely to occur as the situation becomes more complicated and there are loops involved, and how malloc() and free() interact can take multiple paths.

Pointers involved with memory management should always be pointing at 0 or at allocated memory. Anything else is just asking for trouble. Therefore, I deem any direct use of free() dangerous, as it doesn't set the pointer to 0.

So if free() is considered harmful, what should one use?

In C++, I recommend the following:

static inline void insane_free(void *&p)
{
  free(p);
  p = 0;
}

This insane_free() is now a drop in replacement for free(), and can be used instead. (Since C++ programs normally use new and delete/delete[] instead, I leave it as an exercise to the reader how to work with those.)

However, C doesn't support direct references. One can pass a pointer by a pointer to accomplish similar results, but that becomes clunky and is not a drop in replacement. So in C, I recommend the following:
#define insane_free(p) { free(p); p = 0; }
It makes use of the preprocessor, so some may consider it messy, but it can be used wherever free() currently is. One could also name the macro free in order to automatically replace existing code, but it's best not to program that way, as you begin to rely on these semantics. This in turn means someone copying your code may think a call to free() is the normal free() and not realize something special is occurring when they copy it elsewhere without the macro.

Correct usage in simple cases is then:
void *p = malloc(size);
if (p)
{
  ... work with p ...
  insane_free(p);
}
else
{
  ... handle error scenario ...
}
If you think using a wrapper macro or function is overkill, and just always manually assigning the pointer to 0 after freeing is the way to go, consider that it's unwieldy to constantly do so, and you may forget to. If the above technique was always used, all use after free bugs would never have occurred in the first place.

Something else to be aware of is that there's nothing wrong with calling free(0). However, calling free() upon a pointer which is not null and not pointing to allocated memory is forbidden and will crash your program. So stick to the advice here, and you may just find memory management became significantly easier.

If all this talk of pointers is beyond you, consider acquiring Understanding and Using C Pointers.

sprintf() and asprintf()

If you do a lot of C programming, at some point, you probably have used the sprintf() function, or its safer counterpart snprintf().

These two functions sprintf() and snprintf() act like printf(), but instead of printing to the standard output, they print to a fixed-length string buffer. Now a fixed-length string buffer is great and all, but what if you wanted something which automatically allocated the amount it needed?

Enter asprintf(), a function which acts like sprintf(), but is auto-allocating, and needs no buffer supplied. This function was invented ages ago by GLIBC, shortly thereafter copied to the modern BSDs, and found its way further still into all sorts of libraries, although is not yet ubiquitous.

Let's compare the prototypes of the two:
int sprintf(char *buffer, const char *format, ...); 
int asprintf(char **ret, const char *format, ...);
The sanest approach would have been for a function like asprintf() to have the prototype of:
char *asprintf(const char *format, ...);
But its creators wanted to make it act like sprintf(), and its design can also be potentially more useful.

Instead of passing asprintf() a buffer, a pointer to a variable of type char * needs to be passed, like so:
char *buffer;
asprintf(&buffer, ...whatever...);
Now how asprintf() actually works is no big secret. The C99 standard specified that snprintf() upon failure should return the amount of characters that would be needed to contain its output. Which means that conceptually something along the following lines would be all that asprintf() needs to do:
char *buffer = malloc(snprintf(0, 0, format, data...)+1);
sprintf(buffer, format, data...);
Of course though, the above taken verbatim would be incorrect, because it mistakenly assumes that nothing can go wrong, such as the malloc() or snprintf() failing.

First let's better understand what the *printf() functions return. Upon success, they return the amount of characters written to the string (which does not include the trailing null byte). Or in other words, the return value is equivalent to calling strlen() on the data being output, which can save you needing to use a strlen() call with sprintf() or similar functions for certain scenarios. Upon failure, for whatever reason, the return is -1. Of course there's the above mentioned exception to this with snprintf(), where the amount of characters needed to contain the output would be returned instead. If during the output, the size overflows (exceeds INT_MAX), many implementations will return a large negative value (failure with snprintf(), or success with all the functions).

Like the other functions, asprintf() also returns an integer of the nature described above. Which means working with asprintf() should go something like this:
char *buffer;
if (asprintf(&buffer, ...whatever...) != -1)
{
  do_whatever(buffer);
  insane_free(buffer);
}
However, unlike the other functions, asprintf() has a second return value, its first argument, or what the function sees as *ret. To comply with the memory management discussion above, this should also be set to 0 upon failure. Unfortunately, many popular implementations, including those in GLIBC and MinGW fail to do so.

Since I develop with the above systems, and I'm using asprintf() in loops with multiple paths, it becomes unwieldy to need to pass around the buffer and the returned int, so I'd of course want saner semantics which don't leave dangling pointers in my program.

In order to correct such mistakes, I would need to take code from elsewhere, or whip up my own function. Now I find developing functions such as these to be relatively simple, but even so, I always go to check other implementations to see if there's any important points I'm missing before I go implement one. Maybe, I'll even find one which meets my standards with a decent license which I can just copy verbatim.

In researching this, to my shock and horror, I came across implementations which properly ensure that *ret is set to 0 upon failure, but the returned int may be undefined in certain cases. That some of the most popular implementations get one half wrong, and that some of the less popular get the other half wrong is just downright terrifying. This means that there isn't any necessarily portable way to check for failure with the different implementations. I certainly was not expecting that, but with the amount of horrible code out there, I guess I really shouldn't be surprised anymore.

Also in the course of research, besides finding many implementations taking a non-portable approach, many have problems in all sorts of edge cases. Such as mishandling overflow, or not realizing that two successive calls to a *printf() function with the same data may not necessarily yield the same results. Some try to calculate the length needed with some extra logic and only call sprintf() once, but this logic may not be portable, or always needs updating as new types are added to the format string as standards progress, or the C library decided to offer new features. Some of the mistakes I found seem to be due to expecting a certain implementation of underlying functions, and then later the underlying functions were altered, or the code was copied verbatim to another library, without noting the underlying functions acted differently.

So, once again, I'm finding myself needing to supply the world with more usable implementations.

Let's dive into how to implement asprinf().

Every one of these kind of functions actually has two variants, the regular which takes an unlimited amount of arguments, and the v variants which take a va_list (defined in stdarg.h) after the format argument instead. These va_lists are what ... gets turned into after use, and in fact, every non-v *printf() function is actually wrapped to a counterpart v*printf() function. This makes implementing asprintf() itself quite straight forward:



To fix the aforementioned return problems, one could also easily throw in here a check upon the correct return variable used in the underlying vasprintf() implementation and use it to set the other. However, that's not a very portable fix, and the underlying implementation of vasprintf() can have other issues as described above.

A straight forward implementation of vasprintf() would be:



As long as you have a proper C99 implementation of stdarg.h and vsnprintf(), you should be good to go. However, some systems may have vsnprintf() but not va_copy(). The va_copy() macro is needed because a va_list may not just be a simple object, but have handles to elsewhere, and a deep copy is needed. Since vsnprintf() being passed the original va_list may modify its contents, a copy is needed because the function is called twice.

Microsoft Visual C++ (MSVC, or Microsoft Vs. C++ as I like to think of it) up until the latest versions has utterly lacked va_copy(). This and several other  systems that lack it though usually have simple va_lists that can be shallow copied. To gain compatibility with them, simply employ:


#ifndef va_copy 
#define va_copy(dest, src) dest = src 
#endif

Be warned though that if your system lacks va_copy(), and a deep copy is required, using the above is a recipe for disaster.

Once we're dealing with systems where shallow copy works though, the following works just as well, as vsnprintf() will be copying the va_list it receives and won't be modifying other data.



Before we go further, there's two points I'd like to make.
  • Some implementations of vsnprintf() are wrong, and always return -1 upon failure, not the size that would've been needed. On such systems, another approach will need to be taken to calculate the length required, and the implementations here of vasprintf() (and by extension asprintf()) will just always return -1 and *ret (or *strp) will be 0.
  • The code if ((r < 0) || (r > size)) could instead be if (r != size), more on that later.
Now on Windows, vsnprintf() always returns -1 upon failure, in violation of the C99 standard. However, in violation of Microsoft's own specifications, and undocumented, I found that vsnprintf() with the first two parameters being passed 0 as in the above code actually works correctly. It's only when you're passing data there that the Windows implementation violates the spec. But in any case, relying on undocumented behavior is never a good idea.

On certain versions of MinGW, if __USE_MINGW_ANSI_STDIO is defined before stdio.h is included, it'll cause the broken Windows *printf() functions to be replaced with C99 standards compliant ones.

In any case though, Windows actually provides a different function to retrieve the needed length, _vscprintf(). A simple implementation using it would be:



This however makes the mistake of assuming that vsnprintf() is implemented incorrectly as it currently is with MSVC. Meaning this will break if Microsoft ever fixes the function, or you're using MinGW with __USE_MINGW_ANSI_STDIO. So better to use:



Lastly, let me return to that second point from earlier. The vsnprintf() function call the second time may fail because the system ran out of memory to perform its activities once the call to malloc() succeeds, or something else happens on the system to cause it to fail. But also, in a multi-threaded program, the various arguments being passed could have their data change between the two calls.

Now if you're calling functions while another thread is modifying the same variables you're passing to said function, you're just asking for trouble. Personally, I think that all the final check should do is ensure that r is equal to size, and if not, something went wrong, free the data (with insane_free() of course), and set r to -1. However, any value between 0 and size (inclusive), even when not equal to size means the call succeeded for some definition of success, which the above implementations all allow for (except where not possible Microsoft). Based on this idea, several of the implementations I looked at constantly loop while vsnprintf() continues to indicate that the buffer needs to be larger. Therefore, I'll provide such an implementation as well:



Like the first implementation, if all you lacked was va_copy(), and shallow copy is fine, it's easy to get this to work on your platform as described above. But if vsnprintf() isn't implemented correctly (hello MSVC), this will always fail.

All the code here including the appropriate headers, along with notes and usage examples are all packed up and ready to use on my asprintf() implementation website. Between everything offered, you should hopefully find something that works well for you, and is better than what your platform provides, or alternative junk out there.

As always, I'm only human, so if you found any bugs, please inform me.

263 comments:

  1. Code as images? I thought that you were better than that.

    ReplyDelete
  2. Hi henke37,

    As part of the continual degrade of Blogger, it is now very difficult to post code into posts, especially if they contain < or >.

    As a new problem with Blogger that I saw for the first time writing this article, formatting now also randomly changes.

    You'll notice the #ifdef code has weird spacing which I couldn't fix without it degrading into something worse.

    After trying half a dozen times to get the large code samples to display correctly, I gave up and resorted to images.

    In any case, I link to a location you can get the code as text in a tidy package.

    Due to the issues here though, I probably will one of these days take my articles to my own server, with my own blogging software which I can properly manage instead of the nightmare Blogger has become.

    ReplyDelete
  3. Interesting article. I never considered that variable arguments in C might require a deep copy.

    Would you happen to know of a scenario where this is the case?

    ReplyDelete
  4. Offhand, Linux on AMD64 requires it.

    ReplyDelete
  5. To me all this says to use C++ strings wherever possible.

    ReplyDelete
  6. Dear insane coder,

    this page of yours - http://asprintf.insanecoding.org/ - that I found on a DuckDuckGo search did not contain any contact information and neither did the blogspot.com blog of yours. See http://www.shlomifish.org/meta/FAQ/#obscure_email_addr . As a result I have to resort to posting a comment here that there's a typo there - "aquire" should be "acquire" - see https://en.wiktionary.org/wiki/aquire . Please correct it and please include some form of contact information on your web-sites.

    Regards,

    -- Shlomi Fish

    ReplyDelete
  7. Hello Shlomi

    Thank you for catching that typo, it has been fixed.

    The reason I do not include contact information is not because of spam (although spam is an issue), but simply because I don't like being bothered. Most of my readers know how to get a hold of me on IRC (FreeNode). If there's something in particular you'd like to discuss, I can contact you ;-)

    ReplyDelete
  8. When you read about so-and-so celebrity's "rehab that didn't work the first time" you seldom get details about the program. Was it a faith-based 12-step program, or another type of approach Did the person just have drug detox but no rehab How long did they stay with the program Was it in-patient or out-patient? There are many other variables.
    inspirational quotes for addiction
    rehab quotes


    ReplyDelete
  9. This is one of the new internet based life stages that put in it's absolute best effort to after YesPornPlease's dirty tricks. Other than offering an option for
    YesPornPlease clients, it is additionally giving control. I mean over; who sees your substance, offers or remarks posts, offering settled remark strings, boycotting
    and private informing just as client made networks. While it appears to be excessively much for them to eat up, my anxiety is having porn YesPornPlease style, and in the event that they have that, at that point we are acceptable!
    yespornplease

    ReplyDelete
  10. Benificial for Ibps po, Clerk, SBI clerk, PO, RRB PO, Cler and Other Competitive Examination
    English Comprehension 2020

    ReplyDelete
  11. I am looking for and I love to post a comment that provide on this site has helped me greatly. 토토사이트

    ReplyDelete
  12. Thank you. I'll be back every day. Guess I will just bookmark this site. 토토

    ReplyDelete
  13. You’re incredible! Thank you! The blog is instructive additionally 안전놀이터

    ReplyDelete
  14. I’m gone to inform my오피

    little brother, that he should also pay a quick visit this blog on regular basis to obtain updated from most recent
    news.

    ReplyDelete
  15. Jeongsu-bin, who was sprinting after sending a ground ball from the second baseman at the first base in the fifth inning, 0-1 in the 5th inning at the Jamsil Stadium in Seoul, was forced out and complained of pain only to the side after being forced out. .Doosan coach Kim Tae-hyung subtracted Su-Bin Su and put in 먹튀폴리스

    ReplyDelete

  16. the team when defending at the end of the 5th inning.
    Sue-bin said that on the 17th, a close-up examination was found, and damage to my oblique muscle was found. It is an injury that requires a break for more than 10 days.
    Cats more serious. 토토사이트

    ReplyDelete
  17. 여기 처음 가보는 곳인데 한 번에 다 읽어서 정말 감동이에요.
    배치하다토토사이트

    ReplyDelete
  18. I’m still learning from you, while I’m improving myself. I certainly enjoy reading everything that is written on your website.Keep the posts coming. I loved it!
    먹튀검증

    ReplyDelete
  19. Your website is a gift for me.It contains all the information I want to know.My website Custom Mailer Boxes works just like yours.You must visit it if you like it.

    ReplyDelete
  20. Such a great blog i like it very much thanks for sharing with us.
    custom Eye Shadow packaging

    ReplyDelete
  21. Thanks For Sharing such a wonderfull thought I realy appreciate for 바카라사이트

    ReplyDelete
  22. I really happy found knew this website it very informative.


    https://www.thekingcasino.top


    ReplyDelete
  23. Adapted to new systems and processes well and seeks out training to enhance knowledge, skills and abilities.

    https://www.ophunter.net

    ReplyDelete
  24. Not that I am complaining, but slow loading instances times will sometimes affect your placement in google and could damage your high-quality score if advertising and marketing with Adwords. Well I am adding this RSS to my e-mail and can look out for a lot more of your respective fascinating content.

    스포츠마사지

    ReplyDelete
  25. That is a really good tip especially to those fresh to the blogosphere.

    Simple but very accurate info... Many thanks for sharing this one.

    A must read post! https://www.sportstoto.top

    ReplyDelete
  26. Yeah bookmaking this wasn't a bad determination outstanding post!

    My web site; 풀싸롱

    ReplyDelete

  27. Thanks for sharing a nice article really such a wonderful site
    you have done a great job once more thanks a lot토토

    ReplyDelete
  28. I am always searching online for articles that can help me. There is obviously a lot to know about this. I think you made some good points in Features also. Keep working, great job ! 온라인카지노사이트

    ReplyDelete
  29. I have bookmarked your website because this site contains valuable information in it. I am really happy with articles quality and presentation. Thanks a lot for keeping great stuff. I am very much thankful for this site. 카지노사이트링크

    ReplyDelete
  30. I’m truly enjoying the design and layout of your website. It’s a very easy on the eyes which makes it much more enjoyable for me to come here and visit more often. 바카라사이트닷컴

    ReplyDelete
  31. Concrete damage can be caused by a multitude of things. Resurfacing and refinishing concrete damage can take care of a lot of surface damage and give your concrete surface an instant face lift. Website

    ReplyDelete
  32. This comment has been removed by the author.

    ReplyDelete
  33. Thank you a bunch for sharing this with all of us you actually realize what you are talking about! Bookmarked. Please also seek advice from my site =). We could have a hyperlink change contract between us 먹튀검증사이트

    ReplyDelete
  34. The information you share is great and useful to me and many others. It is closely related to my work and has helped me grow. Here we provide information about. ufa168

    ReplyDelete
  35. Many thanks for the article, I have a lot of spray lining knowledge but always learn something new. Keep up the good work and thank you again. 먹튀사이트

    ReplyDelete
  36. Recently, I have started to read a lot of unique articles on different sites, and I am enjoying that a lot. Although, I must tell you that I still like the articles here a lot. They are also unique in their own way. 먹튀검증업체

    ReplyDelete
  37. Just desire to say your article is as astounding. The clarity in your post is simply great and i can assume you’re an expert on this subject. Well with your permission allow me to grab your RSS feed to keep up to date with forthcoming post. Thanks a million and please carry on the rewarding work. ufabet

    ReplyDelete
  38. I was impressed by your writing. Your writing is impressive. I want to write like you.스포츠토토사이트 I hope you can read my post and let me know what to modify. My writing is in I would like you to visit my blog.

    ReplyDelete
  39. https://www.top-travel.ir/%d8%aa%d8%a7%da%86-%d8%a7%d9%84-%d8%b3%db%8c-%d8%af%db%8c-%d8%a2%db%8c%d9%81%d9%88%d9%86/It is also worth mentioning that repairs and replacement of these parts are also done in this collection. IPhone 7 Plus LCD touch repair

    ReplyDelete
  40. Go to office setup for office.com/setup. Sign In or Create a new Microsoft Account. Or go to
    www.office.com/setup to download
    Officecom which is best for online work.

    ReplyDelete
  41. Use garmin.com/express to update maps and software, sync with Garmin Connect™ and register your device. This desktop software notifies you when updates. Best
    pest control near me but we are professional in this field

    ReplyDelete
  42. Xiaomi Redmi Note 9T brings three rear cameras from its previous generation (48-megapixel f / 1.8 main sensor, 2-megapixel macro f / 2.4 and 2-megapixel f / 2.4) but lost the ultra-wide lens that the phone had . You will not take great quality photos with the Note 9T, but it can meet your needs for everyday use.https://www.8deynews.com/567077/

    ReplyDelete
  43. Just how can you have such abilities? I can not evaluate your abilities yet, yet your writing is outstanding. I thought of my instructions once again. I desire a professional like you to review my writing and also court my writing since I'm truly interested regarding my abilities. 바카라사이트

    ReplyDelete
  44. Wow, you did such an amazing job with such a small space. It looks outstanding! Bathroom Remodel Kansas City

    ReplyDelete
  45. very interesting , good job and thanks for sharing such a good blog.Most Expensive Player in IPL

    ReplyDelete
  46. Yes, this is a good post without any doubts. You really do a great job. I inspired by you. So keep it up!! http://www.losangelespainterservices.com/

    ReplyDelete
  47. Terrific article! That is the type of information that
    are meant to be shared around the net. Shame on the seek engines for
    no longer positioning this post higher! Come on over and
    discuss with my web site . Thank you =)Click Here 오피월드


    2YOUNGYANG

    ReplyDelete
  48. I really like your article, your article is really very interesting
    los locos addams
    venom
    after 3
    venom 2
    venom repelis

    ReplyDelete
  49. This is a very educative and helpful blog post, I really learnt a lot going through it, and I must commend you for this great piece which I consider very useful to me and other readers, please keep it up. 룰렛

    ReplyDelete
  50. This comment has been removed by the author.

    ReplyDelete
  51. I was pretty pleased to uncover this site. I need to to thank you for your time for this particularly wonderful read!! I definitely appreciated every part of it and I have you saved to fav to see new information in your web site.

    야한소설

    ReplyDelete
  52. Hello! I just would like to give you a huge thumbs up for your excellent info you have here on this post. I will be returning to your site for more soon.마사지

    ReplyDelete
  53. Great article 건전마사지. Keep writing such kind of information on your page. I'm really impressed by your blog.

    ReplyDelete
  54. This blog was extremely extraordinary, never observed an incredible blog like this previously. I think I am going to share this to my companions. 토토

    ReplyDelete
  55. 토토사이트 Like!! Really appreciate you sharing this blog post.Really thank you! Keep writing.

    ReplyDelete
  56. 온라인카지노 I love your blog.. very nice colors & theme. Keep working ,splendid job!

    ReplyDelete
  57. 카지노사이트 I am happy that you simply shared this useful information with us. Please stay us up to date like this. Thanks for sharing.


    ReplyDelete
  58. Hello friends, its impressive article regarding educationand entirely defined, keep it up all the time. 토토사이트

    ReplyDelete
  59. Fantastic job. I really enjoyed
    what you had to say, and more than that, how you presented it.
    Too cool! 카지노

    ReplyDelete
  60. Useful info. Lucky me I found your web site by accident,
    and I am shocked why this coincidence didn’t happened earlier!
    I bookmarked it. 토토사이트

    ReplyDelete
  61. 카지노사이트 I just found this blog and have high hopes for it to continue. Keep up the great work, its hard to find good ones. I have added to my favorites. Thank You.

    ReplyDelete
  62. 온라인카지노 Thanks for such a great post and the review, I am totally impressed! Keep stuff like this coming.

    ReplyDelete
  63. Thanks for sharing! I definitely enjoying every little bit of it I have you bookmarked to check out new stuff you post. concrete companies

    ReplyDelete
  64. All purchases made with the app are nonrefundable, and there are no refunds for partially used services.
    Reach out to Match's customer service . Inform them that you have canceled your subscription and want to request a refund. Since Match has over 9.9 million paid users, don't be surprised if you are stuck on hold for a long time
    cancel tinder subscription -
    cancel match subscription -
    tinder com refund
    match refund

    ReplyDelete
  65. HP OfficeJet Pro 8600 is a multi-functional printer that is best for use in offices or homes. It comes with some great features like automatic document feeder and instant ink plan. So, simply you do not ever have to worry about empty ink cartridges. other than this, the automatic document feeder will detect, scan and copy different documents on its own.

    ReplyDelete
  66. You have a real ability for writing unique content. I like how you think and the way you represent your views in this article. www.kcityconcretecontractors.com/

    ReplyDelete
  67. It's always coming. This blog is great Thanks on your marveous posting! Greetings! 토토추천

    ReplyDelete
  68. I can't believe there's a post like this 메이저사이트 but I believe it will help a lot in my country

    ReplyDelete
  69. We are linking to this great post on our website It's cool every day 먹튀폴리스

    ReplyDelete
  70. the information provided! Maintain the good performance of your site. You can also check my article junk, which we undergo incredibly frequently. 토토사이트검증

    ReplyDelete
  71. I have noticed this weblog. Lastly something not some of them hope you will give more information on this topics in your next articles 안전한놀이터

    ReplyDelete
  72. when I feel so down but I will feel better right after checking your blog its very informative and your blog is really good and impressive you made it mice 메이저검증업체

    ReplyDelete
  73. Excellent information on your blog, thank you for taking the time to share with us. Amazing insight you have on this, It's nice to find a website that details so much information about different artists.

    온라인카지노

    ReplyDelete
  74. part of my day because you never know how much you make my day happier and more complete.
    There are even times when I feel so down but I will feel better right after checking your blogs 사설토토사이트

    ReplyDelete
  75. Hey I’m reading this on my iPhone and it looks a ton different than on my computer have you noticed this or is it just my phone ? 안전공원추천

    ReplyDelete
  76. I like your blog. i ma happy to read your 토토검증업체 its very informative and your blog is really good and impressive you made it mice article.

    ReplyDelete
  77. e you saved as a favorite to see new information on your blog. NBA중계

    ReplyDelete

  78. I’d need to examine with you here. Which is not something I often do! I enjoy studying a publish that can make people think. Also, thanks for permitting me to remark!
    สล็อตออนไลน์
    สมัครสล็อต
    เติมเงินสล็อต

    ReplyDelete

  79. Thank you for such a well written article. It’s full of insightful information and entertaining descriptions. Your point of view is the best among many.
    ดาวน์โหลดสล็อต
    โปรโมชั่นสล็อต

    ReplyDelete
  80. I consider something genuinely interesting about your website so I bookmarked
    slotxo
    918kiss

    ReplyDelete
  81. My cousin and I had been debating this topic, he is generally looking to show me wrong. Your view on this is fantastic and exactly how I really think. I just sent him this web site to demonstrate him your point of view
    joker123
    pg slot
    goldenslot

    ReplyDelete
  82. I am curious to find out what blog system you have been working with? I’m experiencing some small security problems with my latest blog and I’d like to find something more safe. Do you have any suggestions?
    สล็อตออนไลน์ มือถือ
    สล็อตออนไลน์ fifa55
    สล็อตออนไลน์ ได้เงินจริง

    ReplyDelete
  83. Please continue this great work and I look forward to more of your awesome blog posts. https://www.pioneerhealthcenter.com/

    ReplyDelete
  84. I am very impressed with your writing안전놀이터추천 I couldn't think of this, but it's amazing! I wrote several posts similar to this one, but please come and see!


    ReplyDelete
  85. I finally found what I was looking for! I'm so happy. 메이저사이트


    ReplyDelete
  86. I saw your article well. You seem to enjoy 바카라사이트 for some reason. We can help you enjoy more fun. Welcome anytime :-)


    ReplyDelete
  87. Family game board: Board games that are suitable for the family and its usual number. Like the game board
    https://azadarmaki.ir/blog/8-reportage/41-%D8%A8%D8%B1%D8%AF%DA%AF%DB%8C%D9%85.html

    ReplyDelete
  88. This is a fabulous post I seen by virtue of offer it. It is genuinely what I expected to see look for in future you will continue subsequent to sharing such an extraordinary post. ufabet

    ReplyDelete
  89. Thanks for sharing! I definitely enjoying every little bit of it I have you bookmarked to check out new stuff you post. pool deck footings

    ReplyDelete
  90. It’s the best time to make a few plans for the long run and it is time to be happy.
    I have read this submit and if I could I desire to counsel you
    few interesting issues or advice. Maybe you could write subsequent articles relating to this article.
    I wish to learn more things about it! 경마사이트

    ReplyDelete
  91. Appreciating the hard work you put into your site and detailed information you present. Wonderful read! 스포츠토토

    ReplyDelete
  92. Thanks for this post!! it was great reading this article!! i would like to know more!! keep in touch and stay connected!! Cheers! 한국야동

    Also feel free to visit may web page check this link 야설

    ReplyDelete
  93. Many thanks for the article, I have a lot of spray lining knowledge but always learn something new. Keep up the good work and thank you again. 온라인바카라


    ReplyDelete
  94. This is a great inspiring article.I am pretty much pleased with your good work.You put really very helpful information. concrete driveway tacoma wa

    ReplyDelete
  95. "I was impressed by your writing. Your writing is impressive. I want to write like you.우리카지노 I hope you can read my post and let me know what to modify. My writing is in I would like you to visit my blog.

    ReplyDelete
  96. Nice post. I learn something totally new and challenging on websites 야한동영상

    Also feel free to visit may webpage check this link
    야설

    ReplyDelete
  97. Excellent blog right here! Additionally your website a lot up very fast! What web host are you the usage of? Can I am getting your affiliate hyperlink on your host? I want my website loaded up as quickly as yours lol 무료야설

    ReplyDelete
  98. Have you ever considered about adding a little bit more than just your articles? 오피헌터

    ReplyDelete
  99. "I mean, what you say is fundamental and all.
    However think about if you added some great images
    or video clips to give your posts more, pop! Your content is excellent but with pics and videos, this website could certainly be one of the best in its field. Awesome blog!"

    타이마사지

    ReplyDelete
  100. The clearness on your post is simply spectacular and that i could assume you are a professional on this subject 룰렛

    ReplyDelete
  101. This is my first time visit here. From the tons of comments on your articles,I guess I am not only one having all the enjoyment right here! concrete contractors detroit, mi

    ReplyDelete
  102. Hello there, I discovered your website by means of Google while looking for a related subject,
    your site came up, it appears great. I have bookmarked it in my google bookmarks 토토사이트

    ReplyDelete
  103. Hello there, I discovered your website by means of Google while looking for a related subject,
    your site came up, it appears great. I have bookmarked it in my google bookmarks 토토

    ReplyDelete
  104. Hi....
    If you use sprintf() or vsprintf() , you need to allocate a buffer first, ... dealing with TONS of integers (e.g large arrays) and memory usage is critical.
    You are also read more How to apply for a Business Loan Online

    ReplyDelete
  105. Great web site. A lot of useful information here. I’m sending it to several friends and also sharing in delicious. And obviously, thanks in your effort! 카지노사이트

    ReplyDelete
  106. you have done a great job. I will definitely dig it and personally recommend to my friends. I am confident they will be benefited from this site 야한동영상

    Also feel free to visit may webpage check this link
    야설

    ReplyDelete
  107. Wow! After all I got a web site from where I can actually get valuable information regarding my study and knowledge. 일본야동

    Also feel free to visit may webpage check this link
    한국야동

    ReplyDelete
  108. Hello friends, its fantastic article about teaching and entirely explained, keep it up all the time. 한국야동닷컴

    Also feel free to visit may webpage check this link
    국산야동

    ReplyDelete
  109. Thanks for sharing. I found a lot of interesting information here. A really good post, very thankful and hopeful that you will write many more posts like this one. 국산야동

    Also feel free to visit may webpage check this link
    야설

    ReplyDelete
  110. Amazing article. Your blog helped me to improve myself in many ways thanks for sharing this kind of wonderful informative blogs in live. I have bookmarked more article from this website. Such a nice blog you are providing. 중국야동넷

    Also feel free to visit may webpage check this link
    야설

    ReplyDelete
  111. We stumbled over here by a different website and thought I might check things out. I like what I see so now i am following you. Look forward to finding out about your web page again. 카지노사이트

    ReplyDelete
  112. Nice information, valuable and excellent design, as share good stuff with good ideas and concepts, lots of great information and inspiration, both of which I need, thanks to offer such a helpful information here. ufabet

    ReplyDelete
  113. Your article in information is really so helpful for new readers. i read this website blogs regularly for gain my knowledge about many topics info 온라인카지노

    ReplyDelete
  114. The Move Me Professional Movers and Packers in Dubai. Contact us today for professional movers in Dubai. Make your move stress-free with The Move Me movers and packers in Dubai, UAE.
    best movers packers dubai
    movers packers dubai

    ReplyDelete
  115. To download the software on your windows pc or hp smart app for mac, you could either first visit the Mac store or Microsoft AppStore and quest for the application OR you can without much of a stretch utilize any of the downloads joins we gave above under the Download and Install header segment to download the application.

    ReplyDelete
  116. Awesome and entertaining article. I like to write a little comment to support you. I just stumbled upon your blog and wanted to say that I have really enjoyed reading your blog posts.
    카지노

    ReplyDelete
  117. Really nice and interesting post. I was looking for this kind of information and enjoyed reading this one. Keep posting. Thanks for sharing. 토토

    ReplyDelete
  118. Impressive! Thanks for giving me an idea to my site. Ill be following your works from now on. Hoping for your success. 바카라사이트

    ReplyDelete
  119. Usually I don't learn post on blogs, however I wish to say that this write-up very compelled me to try and do it! Your writing style has been amazed me. Thank you, quite great post. ufabet

    ReplyDelete
  120. Just bookmarked this blog post as I have found it rather valuable. 스포츠토토

    ReplyDelete
  121. Very rapidly this site will be famous among all blogging visitors, due to it's pleasant articles 파친코사이트

    ReplyDelete
  122. Hurrah! Finally I got a weblog from where I be able to actually take valuable facts concerning my study and knowledge. Feel free to visit my website; 배트맨토토프로

    ReplyDelete
  123. 스포츠토토티비 INTERESTING!! you should upload more☺️ this is really good!!!

    ReplyDelete
  124. 카지노 Way cool! Some very valid points! I appreciate you writing this
    write-up and also the rest of the site is very good.


    ReplyDelete
  125. This is an awesome article, Given such an extraordinary measure of data in it, These sort of articles keeps the customers excitement for the site, and keep sharing more ... favorable circumstances. เว็บแทงบอล

    ReplyDelete
  126. Hello! I could have sworn I've visited this website before but after going through a few of the posts I realized it's new to me. Regardless, I'm certainly delighted I came across it and I'll be bookmarking it and checking back regularly! 토토

    ReplyDelete
  127. Wonderful goods from you, man. I have understand your stuff previous to and you are just extremely fantastic. 파워볼

    ReplyDelete
  128. I like viewing web sites which comprehend the price of delivering the excellent useful resource free of charge. I truly adored reading your posting. 바카라사이트

    ReplyDelete
  129. Youre so cool! I dont suppose Ive read anything similar to this before. So nice to get somebody by incorporating original applying for grants this subject.

    view my link now: 스포츠토토

    ReplyDelete
  130. Very interesting , good job and thanks for sharing such a good blog.tellhappystar

    ReplyDelete
  131. Nice post. I was checking continuously this blog and I’m impressed! Very useful info particularly the last part 🙂 I care for such information much. I was seeking this particular information for a long time. Thank you and best of luck. แอพสล็อตxo

    ReplyDelete
  132. good day what a splendid submit i have come upon and ebsite now, and i truely like your fashion. Thank you a million and please hold up the eff nevertheless is just no longer assist to make every sence in any respect preaching approximately that mather. In reality any technique many thanks further to i had undertakin to look wish in future you'll retain for sharing the sort of extremely good publish . Without a doubt i respect the attempt you made to share the understanding. The topic right here i found become certainly effective to the subject which i used to be learning for a long term 안전놀이터

    ReplyDelete
  133. i have examine most of the articles to your internet site now, and i truly like your fashion of running a blog. I delivered it to my favorites blog web site listing and may be checking returned soon. Please test out my website as properly and let me recognize what you observed. I visit your weblog regularly and propose it to all of folks who wanted to enhance their knowledge effectively. The fashion of writing is high-quality and additionally the content material is pinnacle-notch. Thanks for that shrewdness you offer the readers! I used to be browsing the internet for information and came across your blog. I'm inspired by way of the facts you have got in this weblog. It suggests how well you apprehend this issue. Best aspire to say ones content material can be as exceptional. This readability together with your submit is notable and that i might imagine you’re a guru for this trouble. Outstanding along with your concur allow me to to seize your modern supply to hold changed by using using drawing near blog publish. Thank you lots masses of together with you should pass on the fulfilling get the activity done. You have got overwhelmed your self this time, and i recognize you and hopping for some more informative posts in destiny. Thanks for sharing first rate records to us. I think this is a standout amongst the maximum crucial facts for me. What"s extra, i"m satisfied perusing your article. Be that as it can, have to touch upon a few huge things 스피드키노

    ReplyDelete
  134. notable submit i have to say and thank you for the information. Schooling is absolutely a sticky situation. However, continues to be most of the leading topics of our time. I appreciate your publish and look ahead to greater. i am for the first time right here. I discovered this board and i in finding it simply useful & it helped me out a lot. I'm hoping to present something again and assist others including you helped me. I hope that it doesnt disappoint me as lots as this one. I mean, i comprehend it was my choice to study, but i definitely notion you have got some thing interesting to mention. All i hear is a gaggle of whining approximately something that you may restoration if you werent too busy looking for attention. I've read your article; it is very informative and beneficial for me. I appreciate the treasured information you offer to your articles. Thanks for posting it. Your content material is nothing brief of superb in many ways. I suppose this is engaging and eye-starting material. Thank you a lot for caring about your content and your readers. This is a fantastic article, given so much information in it, these sort of articles keeps the customers interest in the internet site, and keep on sharing extra ... Right good fortune . Extremely good info! I latterly got here throughout your weblog and were studying along. I concept i might go away my first remark. I don’t realize what to mention besides that i've 토토사이트

    ReplyDelete
  135. pretty component of substance. I basically observed your site and in boom money to guarantee that i am getting certainly extremely joyful in account your blog entries. Any manner i can buy on your growth or maybe i satisfaction you get segment to usually speedy. It's best to partake in a assignment for a number of the great websites on the internet. I will suggest this website online! This website online is typically a walk thru for everything of the statistics you wanted about this and did not have a clue who to inquire. Influence here, and moreover you may emphatically reveal it. Goodness, amazing weblog structure! How lengthy have you ever been jogging a blog for? You make publishing content material to a blog look easy. The entire look of your site is notable, as intelligently as the substance! That is an awesome blog. I am virtually happy i've determined this statistics. 승인전화없는 토토사이트

    ReplyDelete
  136. This comment has been removed by the author.

    ReplyDelete
  137. first rate website online. Numerous supportive records here. I'm sending it's some thing however a couple of mates ans likewise partaking in tasty. Surely, thanks to your paintings! Female of alien best paintings you could have completed, this website online is completely fascinating with extraordinary subtleties. Time is god as technique of protecting everything from taking place straightforwardly. A good deal obliged to you for supporting, chic facts. If there must be an occurrence of confrontation, in no way try to decide till you ave heard the other side. That is an excellent tip especially to the ones new to the blogosphere. Short but extremely actual statistics recognize your sharing this one. An unquestionable requirement read article! First-rate installation, i honestly love this website, keep on it 꽁머니

    ReplyDelete
  138. Hi....
    Like any library routine, sprintf and snprintf may or may not allocate memory for internal use. They will not allocate memory for the resulting string. That memory must be allocated somehow by the caller, and its address passed as the first argument.
    You are also read more Derivatives Trading Strategies

    ReplyDelete
  139. C for some is basic and simple but it is very difficult for me. I appreciate this post since I learned about it quickly considering that C is hard for me. I'm Anna from www.cardetailinggrandeprairie.com

    ReplyDelete
  140. I was very pleased to find this web-site.I wished to thanks for your time for this wonderful learn!! I positively having fun with every little bit of it and I’ve you bookmarked to check out new stuff you weblog post. 그래프사이트

    ReplyDelete
  141. NFL.com.activate :-The NFL Network owns and operates the NFL Station, an American sports-based pay-TV system.Activate NFL Games on Your Phone & device’s NFL appAs you are on this website then you are also a football enthusiast, and we understand your dedication and craze for NFL. But did you know that now you can stream NFL games online through various platforms like your Fire TV, Apple Tv, Roku, Xbox, and PS4 as well. Isn’t it amazing? So tighten your seatbelt and stick with this blog to the end, and we will explain every minor detail that you need to know for nfl.com/activate. This is going to be an experience that you never have with the NFL.

    ReplyDelete
  142. We absolutely love your blog and find almost all of your post’s to be just what I’m looking for.
    Simple but very accurate info? Thank you for sharing this one. A must read post!
    Appreciating the hard work you put into your site and detailed information you present.
    Wonderful read! I want to write like you. I hope you can read my post and let me know what to modify.
    My writing is in I would like you to visit my blog토토사이트



    ReplyDelete
  143. Im was the runner-up at the 2020 Masters, the best showing by an Asian golfer at the tournament before Hideki Matsuyama of Japan won it all in 2021.
    메이저놀이터
    토토사이트추천
    토토커뮤니티
    먹튀검증
    보증업체
    토토사이트
    안전놀이터

    ReplyDelete
  144. Here’s a referral link for you Marvel fans who want to watch all episodes of Moon Knight streaming online or you want to download it directly:
    Moon Knight episodio 1 streaming
    Moon Knight episodio 2 streaming
    Moon Knight episodio 3 streaming

    ReplyDelete
  145. Memory management does give me a headache because it oftentimes makes my program misbehave. Your post is indeed worth reading. I'm Anna from www.roofingkamloops.com

    ReplyDelete
  146. I was very pleased to find this web-site. I wanted to thanks for your time for this wonderful read!! I definitely enjoying every little bit of it and I have you bookmarked to check out new stuff you blog post.
    If you are looking for customised employee onboarding kit. Get connected with us for more details.

    ReplyDelete
  147. Our experts are very qualified and highly experienced dedicated academicians renowned for their command over the subject, our Business Law Assignment Help Despite being the best quality service provider out there, our Law Assignment Help is always cheap and affordable.

    ReplyDelete
  148. Our compositions are always counterfeit free and totally in sync with the criteria and expectations of your faculty. Great Assignment Helper furnishes the best My Assignment Help UK to students at very affordable rates.

    ReplyDelete
  149. Your post is very interesting to me. Reading was so much fun. I think the reason reading is fun is because it is a post related to that I am interested in. Articles related to 메이저사이트순위 you are the best. I would like you to write a similar post about !
    xdh

    ReplyDelete
  150. Online slots websites that we don't want you to miss out. Online gambling sites that can really make you money. We can guarantee the income you will earn from our website. And we are ready to provide full service to all customers, no matter what type of gambling game, our camp is ready to serve you 24 hours a day, except holidays, ready to serve you fully. You can make a profit. lots pgslot

    ReplyDelete
  151. and enjoy realistic visuals, graphics, light, color, sound that is a three-dimensional system, even if you are a new player, you can play We also added benefits Come to the players, such as organizing promotions for players who have applied for new With the first deposit and activities to join in the fun throughout the month pgslot

    ReplyDelete
  152. A variety of languages ​​for you to choose to change at your convenience. And when you are ready to apply for membership to go to the real field, you need to first look at the various reasons why you have to play games in our camp. Make rewards, so don't wait. Let's look at the reasons why it is better to choose to play games in this camp. Let's take a look at it. pgslot

    ReplyDelete
  153. Our game camp is an online gambling game that has many games for you to choose to bet with your satisfaction and is also popular. It can be said that it is the number one popular game fever in Thailand that has it all. Certify the world-class organization as the best game camp as well. pgslot

    ReplyDelete
  154. Withdraw from the system automatically, no minimum. Give away, of course, don't brag. Do it for real or play play along now Free machine, no need to go online, play the internet, get free, get free, don't have to come to the game Pure will be even more online. because of online games This website has more beautiful pictures than it is fun to play with. สล็อต

    ReplyDelete
  155. The game still offers a simple way to play in the style of PG Slots. which is considered an advantage Players can go to study the symbols before starting to play at Paytable or anyone who is a wader can start riding without wasting time. This simple and easy play is another charm of the game. บาคาร่า ออนไลน์

    ReplyDelete
  156. Place your bets at a rhythm. If you keep playing the betting game, you will be able to catch the right moment and be able to easily win the prize money from the game, but that you only have to place bets in steps. As it is, it will not be difficult to catch the rhythm of betting anymore, just as you will be able to make good winnings. บาคาร่า ออนไลน์

    ReplyDelete
  157. Slotxo, the best and the hottest betting website at the moment, will be a website that cannot be any other than our website, it is Slotxo that you can use easily via mobile. Supports all platforms, whether it's Android or iOS. Easy to make transactions by yourself using the automatic deposit and withdrawal system. It takes less than 1 minute to deposit and withdraw and is also safe for deposits and withdrawals. Importantly, our website also supports languages, including Thai language. We can use it easily. เติมเกม

    ReplyDelete
  158. aylines, game volatility, and game difficulty. More than the various formulas that old players have suggested because it may work for other people. But it doesn't work for the players themselves, it is possible. good bet For new players is placing be ambbet

    ReplyDelete
  159. Our game also has It offers more than 200 games for everyone. Come to play with each other to the fullest. It is a game that is fun to play, win real money, win a jackpot bonus that is frequent and frequent until everyone likes to come to play. Or come to hunt for various prizes, the best game that has received international standards. slotxo

    ReplyDelete
  160. But this website has been upgraded to 200 Press to receive by yourself. Super easy. No need to deposit. No need to share. Apply immediately. Get it right away. No conditions. Unlimited Free Credit Slots Can be used to increase capital in every game This is a privilege that we would like to offer to all our members. Guarantee that it's really giving away superslot

    ReplyDelete
  161. Play and win real money for sure. And you will get the best value in online gambling as well. AMB website enter. pgslot

    ReplyDelete
  162. from the hottest dealers of all famous brands ready to serve everyone to have fun always excited pgslot

    ReplyDelete
  163. there is cause to believe that somehow the creep of racial bias is contaminating the nomination process. The fact that 94 percent of voting members are white pgslot

    ReplyDelete
  164. buttons. Because if you don't have this spin button, you won't be able to spin the wheel for sure, it's a button that is very important and today we don jili

    ReplyDelete
  165. As we said, there will be differences in outcomes if you are still looking for comparisons. pgslot

    ReplyDelete
  166. The volume of the website does not go through middlemen, satisfaction, easy to play, fast money. บาคาร่า ออนไลน์

    ReplyDelete
  167. This comment has been removed by the author.

    ReplyDelete
  168. As Your Business Insurance Specialists, we are dedicated to providing you with reliable, professional service to meet all your needs. | www.isiwc.org

    Please do visit as well as www.homeinsuranceboiseid.com | www.sacramentoemergencydental.net

    ReplyDelete