1*8f0ba417SAndroid Build Coastguard Worker /*
2*8f0ba417SAndroid Build Coastguard Worker * Copyright (C) 2011 The Android Open Source Project
3*8f0ba417SAndroid Build Coastguard Worker *
4*8f0ba417SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*8f0ba417SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*8f0ba417SAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*8f0ba417SAndroid Build Coastguard Worker *
8*8f0ba417SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*8f0ba417SAndroid Build Coastguard Worker *
10*8f0ba417SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*8f0ba417SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*8f0ba417SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*8f0ba417SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*8f0ba417SAndroid Build Coastguard Worker * limitations under the License.
15*8f0ba417SAndroid Build Coastguard Worker */
16*8f0ba417SAndroid Build Coastguard Worker
17*8f0ba417SAndroid Build Coastguard Worker #include "android-base/stringprintf.h"
18*8f0ba417SAndroid Build Coastguard Worker
19*8f0ba417SAndroid Build Coastguard Worker #include <stdio.h>
20*8f0ba417SAndroid Build Coastguard Worker
21*8f0ba417SAndroid Build Coastguard Worker #include <string>
22*8f0ba417SAndroid Build Coastguard Worker
23*8f0ba417SAndroid Build Coastguard Worker namespace android {
24*8f0ba417SAndroid Build Coastguard Worker namespace base {
25*8f0ba417SAndroid Build Coastguard Worker
StringAppendV(std::string * dst,const char * format,va_list ap)26*8f0ba417SAndroid Build Coastguard Worker void StringAppendV(std::string* dst, const char* format, va_list ap) {
27*8f0ba417SAndroid Build Coastguard Worker // First try with a small fixed size buffer
28*8f0ba417SAndroid Build Coastguard Worker char space[1024] __attribute__((__uninitialized__));
29*8f0ba417SAndroid Build Coastguard Worker
30*8f0ba417SAndroid Build Coastguard Worker // It's possible for methods that use a va_list to invalidate
31*8f0ba417SAndroid Build Coastguard Worker // the data in it upon use. The fix is to make a copy
32*8f0ba417SAndroid Build Coastguard Worker // of the structure before using it and use that copy instead.
33*8f0ba417SAndroid Build Coastguard Worker va_list backup_ap;
34*8f0ba417SAndroid Build Coastguard Worker va_copy(backup_ap, ap);
35*8f0ba417SAndroid Build Coastguard Worker int result = vsnprintf(space, sizeof(space), format, backup_ap);
36*8f0ba417SAndroid Build Coastguard Worker va_end(backup_ap);
37*8f0ba417SAndroid Build Coastguard Worker
38*8f0ba417SAndroid Build Coastguard Worker if (result < static_cast<int>(sizeof(space))) {
39*8f0ba417SAndroid Build Coastguard Worker if (result >= 0) {
40*8f0ba417SAndroid Build Coastguard Worker // Normal case -- everything fit.
41*8f0ba417SAndroid Build Coastguard Worker dst->append(space, result);
42*8f0ba417SAndroid Build Coastguard Worker return;
43*8f0ba417SAndroid Build Coastguard Worker }
44*8f0ba417SAndroid Build Coastguard Worker
45*8f0ba417SAndroid Build Coastguard Worker if (result < 0) {
46*8f0ba417SAndroid Build Coastguard Worker // Just an error.
47*8f0ba417SAndroid Build Coastguard Worker return;
48*8f0ba417SAndroid Build Coastguard Worker }
49*8f0ba417SAndroid Build Coastguard Worker }
50*8f0ba417SAndroid Build Coastguard Worker
51*8f0ba417SAndroid Build Coastguard Worker // Increase the buffer size to the size requested by vsnprintf,
52*8f0ba417SAndroid Build Coastguard Worker // plus one for the closing \0.
53*8f0ba417SAndroid Build Coastguard Worker int length = result + 1;
54*8f0ba417SAndroid Build Coastguard Worker char* buf = new char[length];
55*8f0ba417SAndroid Build Coastguard Worker
56*8f0ba417SAndroid Build Coastguard Worker // Restore the va_list before we use it again
57*8f0ba417SAndroid Build Coastguard Worker va_copy(backup_ap, ap);
58*8f0ba417SAndroid Build Coastguard Worker result = vsnprintf(buf, length, format, backup_ap);
59*8f0ba417SAndroid Build Coastguard Worker va_end(backup_ap);
60*8f0ba417SAndroid Build Coastguard Worker
61*8f0ba417SAndroid Build Coastguard Worker if (result >= 0 && result < length) {
62*8f0ba417SAndroid Build Coastguard Worker // It fit
63*8f0ba417SAndroid Build Coastguard Worker dst->append(buf, result);
64*8f0ba417SAndroid Build Coastguard Worker }
65*8f0ba417SAndroid Build Coastguard Worker delete[] buf;
66*8f0ba417SAndroid Build Coastguard Worker }
67*8f0ba417SAndroid Build Coastguard Worker
StringPrintf(const char * fmt,...)68*8f0ba417SAndroid Build Coastguard Worker std::string StringPrintf(const char* fmt, ...) {
69*8f0ba417SAndroid Build Coastguard Worker va_list ap;
70*8f0ba417SAndroid Build Coastguard Worker va_start(ap, fmt);
71*8f0ba417SAndroid Build Coastguard Worker std::string result;
72*8f0ba417SAndroid Build Coastguard Worker StringAppendV(&result, fmt, ap);
73*8f0ba417SAndroid Build Coastguard Worker va_end(ap);
74*8f0ba417SAndroid Build Coastguard Worker return result;
75*8f0ba417SAndroid Build Coastguard Worker }
76*8f0ba417SAndroid Build Coastguard Worker
StringAppendF(std::string * dst,const char * format,...)77*8f0ba417SAndroid Build Coastguard Worker void StringAppendF(std::string* dst, const char* format, ...) {
78*8f0ba417SAndroid Build Coastguard Worker va_list ap;
79*8f0ba417SAndroid Build Coastguard Worker va_start(ap, format);
80*8f0ba417SAndroid Build Coastguard Worker StringAppendV(dst, format, ap);
81*8f0ba417SAndroid Build Coastguard Worker va_end(ap);
82*8f0ba417SAndroid Build Coastguard Worker }
83*8f0ba417SAndroid Build Coastguard Worker
84*8f0ba417SAndroid Build Coastguard Worker } // namespace base
85*8f0ba417SAndroid Build Coastguard Worker } // namespace android
86