| // Copyright 2014 The Android Open Source Project |
| // |
| // This software is licensed under the terms of the GNU General Public |
| // License version 2, as published by the Free Software Foundation, and |
| // may be copied, distributed, and modified under those terms. |
| // |
| // This program is distributed in the hope that it will be useful, |
| // but WITHOUT ANY WARRANTY; without even the implied warranty of |
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| // GNU General Public License for more details. |
| |
| #include "android/base/StringFormat.h" |
| |
| #include <stdio.h> |
| |
| namespace android { |
| namespace base { |
| |
| String StringFormat(const char* format, ...) { |
| va_list args; |
| va_start(args, format); |
| String result = StringFormatWithArgs(format, args); |
| va_end(args); |
| return result; |
| } |
| |
| String StringFormatWithArgs(const char* format, va_list args) { |
| String result; |
| StringAppendFormatWithArgs(&result, format, args); |
| return result; |
| } |
| |
| void StringAppendFormat(String* string, const char* format, ...) { |
| va_list args; |
| va_start(args, format); |
| StringAppendFormatWithArgs(string, format, args); |
| va_end(args); |
| } |
| |
| void StringAppendFormatWithArgs(String* string, |
| const char* format, |
| va_list args) { |
| size_t cur_size = string->size(); |
| size_t extra = 0; |
| for (;;) { |
| va_list args2; |
| va_copy(args2, args); |
| int ret = vsnprintf(&(*string)[cur_size], extra, format, args2); |
| va_end(args2); |
| |
| if (ret == 0) { |
| // Nothing to do here. |
| break; |
| } |
| |
| if (ret > 0) { |
| size_t ret_sz = static_cast<size_t>(ret); |
| if (extra == 0) { |
| // First pass, resize the string and try again. |
| extra = ret_sz + 1; |
| string->resize(cur_size + extra); |
| continue; |
| } |
| if (ret_sz < extra) { |
| // Second pass or later, success! |
| string->resize(cur_size + ret_sz); |
| return; |
| } |
| } |
| |
| // NOTE: The MSVCRT.DLL implementation of snprintf() is broken and |
| // will return -1 in case of truncation. This code path is taken |
| // when this happens, or when |ret_sz| is equal or larger than |
| // |extra|. Grow the buffer to allow for more room, then try again. |
| extra += (extra >> 1) + 32; |
| string->resize(cur_size + extra); |
| } |
| } |
| |
| |
| } // namespace base |
| } // namespace android |