xref: /aosp_15_r20/system/libbase/stringprintf.cpp (revision 8f0ba417480079999ba552f1087ae592091b9d02)
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