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.

99 comments:

henke37 said...

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

insane coder said...

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.

Unknown said...

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?

insane coder said...

Offhand, Linux on AMD64 requires it.

Jon Schneider said...

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

Shlomi Fish said...

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

insane coder said...

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 ;-)

Jatin Sethi said...

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


babloo said...

happy anniversary di and jiju
happy anniversary uncle aunty

Jatin Sethi said...

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

MBBS in Philippines said...

UV GULLAS COLLEGE OF MEDICINE is one of Top Medical College in Philippines in Cebu city. International students have the oppertunity to study medicine in phillipines at affordable cost and world class University. The college has successful alumni who have achieved well in the fields of law, business, politics, academe, medicine, sports and other endeavors. At University of the Visayas, we prepare students for a global competition.

Direct MBBS Admissions Open: 2020-21
Mobile No: +91 90329 55688
Apply Now: https://www.careerplus.org.in/philippines-medical-college/uv-gullas-college-of-medicine

Hindifly said...

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

Hindifly said...

What is Bitcoin & How to Buy Bitcoin

Hindifly said...

What is NPA ( Non- Performing Assets )

Hindifly said...

Thanks for shairing this information
Statutory and Regulatory Provisions

nirjonniyaz said...

토토사이트검증 토토 먹튀 검증 저희 먹튀커머스 는 2016년 5월 부터 지금까지 먹튀커머스 를 믿고 방문 해주시는 유저 분들을 위해 더이상 먹튀 없는 공정한 배팅 문화 를 만들기 위해서 항상 노력하고 유저분들에 소리에 귀를 기울리는 NO.1 먹튀검증 커뮤니티 입니다. 또한 먹튀커머스 에서는 무분별한 배팅사이트 들을 일방적으로 추천 하지 않고 철저한 검수 작업을 토대로 사전에 먹튀 사고가 발생 안되게끔 유저 분들 에게 추천하는 만큼 저희측에 등록 되어 있는 배팅사이트 내에서 혹여 먹튀가 발생 한다면 오로지 그책임은 저희 먹튀커머스 에 있음을 알려 드립니다. / 먹튀 검증

michael said...

yespornplease
yespornplease

Author said...

its a great article !!!!!!!!! JobAlert247 A Job Alert Website


MovieRulz Best Movie Download Website

MTTMT said...

Hello, I really enjoyed reading your post. Actually I'm doing something similar to yours. You and I have a lot in common. Like you, I have posted a lot of articles on my website. If you are interested, please visit and read. Thank you. Have a nice day. 토토사이트

메이저사이트 said...

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

cxzcxc said...

Thanks for your marvelous posting! I actually enjoyed reading it, you could be
a great author.I will remember to bookmark your blog and will먹튀검증


eventually come back from now on. I want to encourage you to continue your great
writing, have a nice weekend!

GOGOLIVE said...

Hello ! I am the one who writes posts on these topics온라인포커 I would like to write an article based on your article. When can I ask for a review?

Packaging Printers said...

Thanks for your nice post I really like it and appreciate it. My work is about Custom Vape Cartridge Boxes. If you need perfect quality boxes then you can visit our website.

Sarah Sadie said...

Thanks for your nice post I really like it and appreciate it.
free netflix accounts flixarena
purse string wrinkles
karmically linked zodiac signs

보증업체 said...

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

보증업체 said...

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

Unknown said...

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.

Unknown said...

Nice Blog. Thanks for sharing with us. Such amazing information.

Only Blog

Guest Blogger

Guest Blogging Site

Guest Blogging Website

Guest Posting Site

토토사이트 said...

Thank you and thank you so much. I'll share you useful information.토토사이트

좋은사이토토 said...

Hello
Thank you for giving me useful information.
Please keep posting good information in the future
I will visit you often. Thank you.
I am also running the site. 메이저놀이터 This is a related site, so please visit once.
Have a niceday!

15321 said...

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 먹튀폴리스

15321 said...


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. 토토사이트

Unknown said...

Nice Blog. Thanks for sharing with us. Such amazing information.

Only Blog

Guest Blogger

Guest Blogging Site

Guest Blogging Website

Guest Posting Site

RK said...

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

leene333 said...

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!
먹튀검증

Digital Vishnu said...

This is incredibly useful information!! Excellent work. All is very fascinating to learn and simple to grasp. Thanks for sharing such great info. Keep Post These kinds of Articles in the future.

Digital Marketing Course in Coimbatore
Digital Marketing Course Training in Tirupur
Digital Marketing Course Training in Madurai
Digital Marketing Course Training in Theni
Digital Marketing Training in Coimbatore

Boris Thomson said...

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.

먹튀검증 said...

We have a site for sports games. 토토사이트 You must be interested. 먹튀사이트

custom printed corrugated boxes said...

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

GOGOLIVE said...

I've been looking for photos and articles on this topic over the past few days due to a school assignment, and I'm really happy to find a post with the material I was looking for! I bookmark and will come often! Thanks :D 바둑이사이트

Unknown said...

As I am looking at your writing, 먹튀검증사이트 I regret being unable to do outdoor activities due to Corona 19, and I miss my old daily life. If you also miss the daily life of those days, would you please visit my site once? My site is a site where I post about photos and daily life when I was free.

먹튀검증 said...

Thanks for writing this awesome 메이저사이트. I'm a long time reader but I've never been compelled to leave a comment. I subscribed to your blog and shared this on my Facebook. Thanks again for a great article!

Uknown said...

ou need to be a part of a contest for one of the highest quality sites on the 토토사이트. I most certainly will recommend this site!

GOGOLIVE said...

It's really great. Thank you for providing a quality article. There is something you might be interested in. Do you know 온라인홀덤 ?

Unknown said...

From some point on, I am preparing to build my site while browsing various sites. It is now somewhat completed. If you are interested, please come to play with 토토사이트 !

Unknown said...

Good day! This post could not be written any better! Reading this post reminds me of my previous room mate! He always kept chatting about this. I will forward this page to him. Pretty sure he will have a good read. Thanks for sharing. 안전사이트

Unknown said...

Your explanation is organized very easy to understand!!! I understood at once. Could you please post about 사설토토 ?? Please!!

rsagar said...

This is a very nice one and gives in-depth information. I am really happy with the quality and presentation of the article. I’d really like to appreciate the efforts you get with writing this post. Thanks for sharing.
Python Course in Bangalore

Unknown said...

I've been looking for photos and articles on this topic over the past few days due to a school assignment, 메이저놀이터순위 and I'm really happy to find a post with the material I was looking for! I bookmark and will come often! Thanks :D

najina said...

Película Clifford, el gran perro rojo
El paraíso del amor Película Completa
Canción de cuna Película Completa
Cool world Película Completa
Aquaslash Película Completa
Victor Frankenstein Película Completa
Glass Película Completa
Película Rápidos y furiosos 9
Riesgo Bajo Cero Película Completa
Boss Level Película Completa

najina said...

I really like your article, and I'm impressed High Society película completa Bad Sister película completa Luca película completa Rapido y Furioso 9 película completa Umibe no Etranger película completa Perros Callejeros película completa Closer película completa Demon slayer el tren infinito película completa Cruella película completa El conjuro 3 película completa 

casinosite.one said...

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

casinosite.one said...

I really happy found knew this website it very informative.


https://www.thekingcasino.top


yasul.top said...

Hello there and thank you for your information – I've definitely picked up anything new from right here. I did however expertise a few technical points using this website, since I experienced to reload the site a lot of times previous to I could get it to load properly. I had been wondering if your web host is OK?

Look into my page: https://www.yasul.top

ophunter.net said...

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

https://www.ophunter.net

massage.blue said...

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.

스포츠마사지

gunma.top said...

I want to to thank you for this great read!! I certainly enjoyed every little bit of it. I have you bookmarked to check out new stuff you post… 출장마사지

Unknown said...

As I am looking at your writing, 먹튀검증사이트 I regret being unable to do outdoor activities due to Corona 19, and I miss my old daily life. If you also miss the daily life of those days, would you please visit my site once? My site is a site where I post about photos and daily life when I was free.

ádccdvfvsd said...

Hello, I read the post well. 안전놀이터추천 It's a really interesting topic and it has helped me a lot. In fact, I also run a website with similar content to your posting. Please visit once

najina said...

pepita la pistolera pelicula completa
medianoche en paris pelicula completa
black widow pelicula completa
encanto pelicula completa
la viuda negra marvel pelicula completa
viuda negra pelicula completa
black dog pelicula completa
harry potter y la piedra filosofal pelicula completa
date movie pelicula completa
esperando la carroza pelicula completa




reinas o reyes pelicula completa
whiplash pelicula completa
tres veces tu pelicula completa
thelma y louise pelicula completa
mi socio 2 pelicula completa
un jefe en pañales 2 pelicula completa
jefe en pañales pelicula completa
coraline y la puerta secreta pelicula completa
amor de gata pelicula completa
fast and furious 9 pelicula completa

najina said...

the woman pelicula completa
mujeres al ataque pelicula completa
mi otra yo pelicula completa
buscando a nemo pelicula completa
cruella pelicula completa
el conjuro 3 pelicula completa
luca pelicula completa
rapido y furioso 9 pelicula completa
Charlie y la fabrica de chocolates Película Completa
venom pelicula completa

intocable pelicula completa
below her mouth pelicula completa
need for speed pelicula completa
50 sombras más oscuras pelicula completa
spirit pelicula completa
medianoche en paris pelicula completa
diabolica tentacion pelicula completa
fuera del cielo pelicula completa
barbie y las 12 princesas bailarinas pelicula completa
la calle del terror parte 2 pelicula completa

Unknown said...

Your explanation is organized very easy to understand!!! I understood at once. Could you please post about 사설토토 ?? Please!!

먹튀신고 said...

It's very interesting. And it's fun. This is a timeless article. I also write articles related to , and I run a community related to 메이저놀이터. For more information, please feel free to visit !!

ádccdvfvsd said...

What a post I've been looking for! I'm very happy to finally read this post. 토토사이트 Thank you very much. Can I refer to your post on my website? Your post touched me a lot and helped me a lot. If you have any questions, please visit my site and read what kind of posts I am posting. I am sure it will be interesting.

bamgosoo said...

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

Crown999 said...

I'm writing on this topic these days, Kèo hay, but I have stopped writing because there is no reference material. Then I accidentally found your article. I can refer to a variety of materials, so I think the work I was preparing will work! Thank you for your efforts.

bamgosoowm said...

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

My web site; 풀싸롱

Unknown said...

Looking at this article, I miss the time when I didn't wear a mask. 먹튀검증업체 Hopefully this corona will end soon. My blog is a blog that mainly posts pictures of daily life before Corona and landscapes at that time. If you want to remember that time again, please visit us.

ádccdvfvsd said...

That's a really impressive new idea! 안전한놀이터 It touched me a lot. I would love to hear your opinion on my site. Please come to the site I run once and leave a comment. Thank you.

Unknown said...

It's really great. Thank you for providing a quality article. There is something you might be interested in. Do you know 토토커뮤니티 ? If you have more questions, please come to my site and check it out!

Unknown said...

Good day! This post could not be written any better! Reading this post reminds me of my previous room mate! He always kept chatting about this. I will forward this page to him. Pretty sure he will have a good read. Thanks for sharing. 안전사이트

1509 said...

Really no matter if someone doesn't be aware of after that its up to other users that they will help, so here it takes place 토토사이트추천.

Unknown said...

Looking at this article, I miss the time when I didn't wear a mask. 먹튀검증업체 Hopefully this corona will end soon. My blog is a blog that mainly posts pictures of daily life before Corona and landscapes at that time. If you want to remember that time again, please visit us.

Unknown said...

I love what you 메이저놀이터 tend to be up too. This kind of clever work and exposure! Keep up the excellent works guys I've incorporated you guys to our blogroll.

mj said...

Workplaces rarely thrive when employees can only work from their desks and WLAN Deployment has been a tremendous boon for workplace productivity. WLAN Deployment

Crown999 said...

I have been looking for articles on these topics for a long time. Kèobóngđá I don't know how grateful you are for posting on this topic. Thank you for the numerous articles on this site, I will subscribe to those links in my bookmarks and visit them often. Have a nice day.

office.com/setup said...

Amazing website, Love it. Great work done. Nice website. Love it. This is really nice.
office setup
Microsoft365.com/setup
Microsoft365.com/setup
ij.start.cannon
www.amazon.com/mytv
office.com/setup

yyuuunnnn said...


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

yyuuunnnn said...

I think this is a really good article. You make this information interesting and
engaging. You give readers a lot to think about and I appreciate that kind of writing.먹튀

yyuuunnnn said...

I love this blog, which is why I keep coming back토토

1509 said...

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. 안전놀이터모음

charles said...

www.amazon.com/mytv
primevideo.com/mytv
youtube.com/activate
youtube.com/activate
youtube.com/activate
install aol desktop gold
youtube.com/activate
fubo.tv/connect
roku.com/link
amazon.com/mytv enter code
amazon.com/mytv enter code
123.hp.com/setup
123.hp.com/setup
123.hp.com/setup
hulu.com/activate
hulu.com/activate
hulu.com/activate
install aol desktop gold
www.amazon.com/mytv
www.amazon.com/mytv
www.primevideo.com/mytv
amazon.com/mytv
www.primevideo.com/mytv
garmin.com/express
xfinity.com/authorize
www.amazon.com/mytv
fubo.tv/connect
fubo.tv/connect
123.hp.com/setup
webroot.com/safe
aol desktop gold download
youtube.com/activate
mcafee.com/activate

1509 said...

Hey there! I could have sworn I’ve been to this website before but after reading through some of the post I realized it’s new to me. Nonetheless, I’m definitely happy I found it and I’ll be book-marking and checking back frequently 안전놀이터추천

Unknown said...

It is my first visit to your blog, and I am very impressed with the articles that you serve. Give adequate knowledge for me. Thank you for sharing useful material. I will be back for the more great post. 먹튀검증사이트 But by chance looking at your post solved my problem! I will leave my blog, so when would you like to visit it?!

1509 said...

Good morning!! I am also blogging with you. In my blog, articles related to are mainly written, and they are usually called 메이저사이트. If you are curious about , please visit!!

Unknown said...

Excellent Blog! I would like to thank for the efforts you have made in writing this post. I am hoping the same best work from you in the future as well. I wanted to thank you for this websites! Thanks for sharing. Great websites 메이저토토추천

1509 said...

It's very interesting. And it's fun. This is a timeless article. I also write articles related to , and I run a community related to 메이저놀이터. For more information, please feel free to visit !!

Unknown said...

Thanks for your marvelous posting! I actually enjoyed reading it, you could be
a great author.I will remember to bookmark your blog and will
eventually come back from now on. I want to encourage you to continue your great
writing, have a nice weekend!

Website:바카라

Unknown said...

Hi! This is my first visit to your blog! We are a team of volunteers and new initiatives in the same niche. Blog gave us useful information to work. You have done an amazing job! 메이저토토사이트 Thank you very much. Can I refer to your post on my website? Your post touched me a lot and helped me a lot. If you have any questions, please visit my site and read what kind of posts I am posting. I am sure it will be interesting.

Unknown said...

This is one very interesting post. I like the way you write and I will bookmark your blog to my favorites. 사설토토사이트

Unknow said...

Thanks for such a fantastic blog. Where else could anyone get that kind of info written in such a perfect way? I have a presentation that I am presently writhing on, and I have been on the look out for such great information. 메이저토토

Crown999 said...

I've been searching for hours on this topic and finally found your post. Tàixỉu, I have read your post and I am very impressed. We prefer your opinion and will visit this site frequently to refer to your opinion. When would you like to visit my site?

bloger said...

I think this is an informative post and it is very beneficial and knowledgeable. Therefore, I would like to thank you for the endeavors that you have made in writing this article. All the content is absolutely well-researched. Thanks스포츠토토사이트

1509 said...

I've been looking for photos and articles on this topic over the past few days due to a school assignment, and I'm really happy to find a post with the material I was looking for! I bookmark and will come often! Thanks :D 먹튀신고

bloger said...

This post is really magnificent. I really like this post. It is one of the best posts that I ve read in a long time. Thanks a lot for this really good post. I really appreciate it 메이저토토

Unknown said...

That's a really impressive new idea! 안전한놀이터 It touched me a lot. I would love to hear your opinion on my site. Please come to the site I run once and leave a comment. Thank you.

Crown999 said...

As I am looking at your writing, Bóng độ I regret being unable to do outdoor activities due to Corona 19, and I miss my old daily life. If you also miss the daily life of those days, would you please visit my site once? My site is a site where I post about photos and daily life when I was free.

Unknown said...

I seriously love your site.. Very nice colors & theme. Did you create this site yourself? Please reply back as I’m trying to create my very own site and would like to learn where you got this from or exactly what the theme is named. Many thanks. 먹튀커뮤니티

bloger said...

Positive site, where did u come up with the information on this posting?I have read a few of the articles on your website now, and I really like your style. Thanks a million and please keep up the effective work. 메이저놀이터추천