1 //
2 //
3 // Copyright 2015 gRPC authors.
4 //
5 // Licensed under the Apache License, Version 2.0 (the "License");
6 // you may not use this file except in compliance with the License.
7 // You may obtain a copy of the License at
8 //
9 //     http://www.apache.org/licenses/LICENSE-2.0
10 //
11 // Unless required by applicable law or agreed to in writing, software
12 // distributed under the License is distributed on an "AS IS" BASIS,
13 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 // See the License for the specific language governing permissions and
15 // limitations under the License.
16 //
17 //
18 
19 #include <grpc/support/port_platform.h>
20 
21 #include "src/core/lib/channel/channel_args.h"
22 
23 #include <limits.h>
24 #include <stdlib.h>
25 #include <string.h>
26 
27 #include <algorithm>
28 #include <initializer_list>
29 #include <map>
30 #include <vector>
31 
32 #include "absl/strings/match.h"
33 #include "absl/strings/str_cat.h"
34 #include "absl/strings/str_format.h"
35 #include "absl/strings/str_join.h"
36 
37 #include <grpc/support/alloc.h>
38 #include <grpc/support/log.h>
39 #include <grpc/support/string_util.h>
40 
41 #include "src/core/lib/gpr/useful.h"
42 #include "src/core/lib/gprpp/crash.h"
43 #include "src/core/lib/gprpp/match.h"
44 
45 namespace grpc_core {
46 
Pointer(void * p,const grpc_arg_pointer_vtable * vtable)47 ChannelArgs::Pointer::Pointer(void* p, const grpc_arg_pointer_vtable* vtable)
48     : p_(p), vtable_(vtable == nullptr ? EmptyVTable() : vtable) {}
49 
Pointer(const Pointer & other)50 ChannelArgs::Pointer::Pointer(const Pointer& other)
51     : p_(other.vtable_->copy(other.p_)), vtable_(other.vtable_) {}
52 
Pointer(Pointer && other)53 ChannelArgs::Pointer::Pointer(Pointer&& other) noexcept
54     : p_(other.p_), vtable_(other.vtable_) {
55   other.p_ = nullptr;
56   other.vtable_ = EmptyVTable();
57 }
58 
EmptyVTable()59 const grpc_arg_pointer_vtable* ChannelArgs::Pointer::EmptyVTable() {
60   static const grpc_arg_pointer_vtable vtable = {
61       // copy
62       [](void* p) { return p; },
63       // destroy
64       [](void*) {},
65       // cmp
66       [](void* p1, void* p2) -> int { return QsortCompare(p1, p2); },
67   };
68   return &vtable;
69 }
70 
71 ChannelArgs::ChannelArgs() = default;
72 ChannelArgs::~ChannelArgs() = default;
73 ChannelArgs::ChannelArgs(const ChannelArgs& other) = default;
74 ChannelArgs& ChannelArgs::operator=(const ChannelArgs& other) = default;
75 ChannelArgs::ChannelArgs(ChannelArgs&& other) noexcept = default;
76 ChannelArgs& ChannelArgs::operator=(ChannelArgs&& other) noexcept = default;
77 
Get(absl::string_view name) const78 const ChannelArgs::Value* ChannelArgs::Get(absl::string_view name) const {
79   return args_.Lookup(name);
80 }
81 
Contains(absl::string_view name) const82 bool ChannelArgs::Contains(absl::string_view name) const {
83   return Get(name) != nullptr;
84 }
85 
operator <(const ChannelArgs & other) const86 bool ChannelArgs::operator<(const ChannelArgs& other) const {
87   return args_ < other.args_;
88 }
89 
operator ==(const ChannelArgs & other) const90 bool ChannelArgs::operator==(const ChannelArgs& other) const {
91   return args_ == other.args_;
92 }
93 
operator !=(const ChannelArgs & other) const94 bool ChannelArgs::operator!=(const ChannelArgs& other) const {
95   return !(*this == other);
96 }
97 
WantMinimalStack() const98 bool ChannelArgs::WantMinimalStack() const {
99   return GetBool(GRPC_ARG_MINIMAL_STACK).value_or(false);
100 }
101 
ChannelArgs(AVL<std::string,Value> args)102 ChannelArgs::ChannelArgs(AVL<std::string, Value> args)
103     : args_(std::move(args)) {}
104 
Set(grpc_arg arg) const105 ChannelArgs ChannelArgs::Set(grpc_arg arg) const {
106   switch (arg.type) {
107     case GRPC_ARG_INTEGER:
108       return Set(arg.key, arg.value.integer);
109     case GRPC_ARG_STRING:
110       if (arg.value.string != nullptr) return Set(arg.key, arg.value.string);
111       return Set(arg.key, "");
112     case GRPC_ARG_POINTER:
113       return Set(arg.key,
114                  Pointer(arg.value.pointer.vtable->copy(arg.value.pointer.p),
115                          arg.value.pointer.vtable));
116   }
117   GPR_UNREACHABLE_CODE(return ChannelArgs());
118 }
119 
FromC(const grpc_channel_args * args)120 ChannelArgs ChannelArgs::FromC(const grpc_channel_args* args) {
121   ChannelArgs result;
122   if (args != nullptr) {
123     for (size_t i = 0; i < args->num_args; i++) {
124       result = result.Set(args->args[i]);
125     }
126   }
127   return result;
128 }
129 
MakeCArg(const char * name) const130 grpc_arg ChannelArgs::Value::MakeCArg(const char* name) const {
131   char* c_name = const_cast<char*>(name);
132   return Match(
133       rep_,
134       [c_name](int i) { return grpc_channel_arg_integer_create(c_name, i); },
135       [c_name](const std::shared_ptr<const std::string>& s) {
136         return grpc_channel_arg_string_create(c_name,
137                                               const_cast<char*>(s->c_str()));
138       },
139       [c_name](const Pointer& p) {
140         return grpc_channel_arg_pointer_create(c_name, p.c_pointer(),
141                                                p.c_vtable());
142       });
143 }
144 
operator <(const Value & rhs) const145 bool ChannelArgs::Value::operator<(const Value& rhs) const {
146   if (rhs.rep_.index() != rep_.index()) return rep_.index() < rhs.rep_.index();
147   switch (rep_.index()) {
148     case 0:
149       return absl::get<int>(rep_) < absl::get<int>(rhs.rep_);
150     case 1:
151       return *absl::get<std::shared_ptr<const std::string>>(rep_) <
152              *absl::get<std::shared_ptr<const std::string>>(rhs.rep_);
153     case 2:
154       return absl::get<Pointer>(rep_) < absl::get<Pointer>(rhs.rep_);
155     default:
156       Crash("unreachable");
157   }
158 }
159 
operator ==(const Value & rhs) const160 bool ChannelArgs::Value::operator==(const Value& rhs) const {
161   if (rhs.rep_.index() != rep_.index()) return false;
162   switch (rep_.index()) {
163     case 0:
164       return absl::get<int>(rep_) == absl::get<int>(rhs.rep_);
165     case 1:
166       return *absl::get<std::shared_ptr<const std::string>>(rep_) ==
167              *absl::get<std::shared_ptr<const std::string>>(rhs.rep_);
168     case 2:
169       return absl::get<Pointer>(rep_) == absl::get<Pointer>(rhs.rep_);
170     default:
171       Crash("unreachable");
172   }
173 }
174 
ToC() const175 ChannelArgs::CPtr ChannelArgs::ToC() const {
176   std::vector<grpc_arg> c_args;
177   args_.ForEach([&c_args](const std::string& key, const Value& value) {
178     c_args.push_back(value.MakeCArg(key.c_str()));
179   });
180   return CPtr(static_cast<const grpc_channel_args*>(
181       grpc_channel_args_copy_and_add(nullptr, c_args.data(), c_args.size())));
182 }
183 
Set(absl::string_view name,Pointer value) const184 ChannelArgs ChannelArgs::Set(absl::string_view name, Pointer value) const {
185   return Set(name, Value(std::move(value)));
186 }
187 
Set(absl::string_view name,int value) const188 ChannelArgs ChannelArgs::Set(absl::string_view name, int value) const {
189   return Set(name, Value(value));
190 }
191 
Set(absl::string_view key,Value value) const192 ChannelArgs ChannelArgs::Set(absl::string_view key, Value value) const {
193   return ChannelArgs(args_.Add(std::string(key), std::move(value)));
194 }
195 
Set(absl::string_view key,absl::string_view value) const196 ChannelArgs ChannelArgs::Set(absl::string_view key,
197                              absl::string_view value) const {
198   return Set(key, std::string(value));
199 }
200 
Set(absl::string_view key,const char * value) const201 ChannelArgs ChannelArgs::Set(absl::string_view key, const char* value) const {
202   return Set(key, std::string(value));
203 }
204 
Set(absl::string_view key,std::string value) const205 ChannelArgs ChannelArgs::Set(absl::string_view key, std::string value) const {
206   return Set(key, Value(std::move(value)));
207 }
208 
Remove(absl::string_view key) const209 ChannelArgs ChannelArgs::Remove(absl::string_view key) const {
210   return ChannelArgs(args_.Remove(key));
211 }
212 
GetInt(absl::string_view name) const213 absl::optional<int> ChannelArgs::GetInt(absl::string_view name) const {
214   auto* v = Get(name);
215   if (v == nullptr) return absl::nullopt;
216   const auto* i = v->GetIfInt();
217   if (i == nullptr) return absl::nullopt;
218   return *i;
219 }
220 
GetDurationFromIntMillis(absl::string_view name) const221 absl::optional<Duration> ChannelArgs::GetDurationFromIntMillis(
222     absl::string_view name) const {
223   auto ms = GetInt(name);
224   if (!ms.has_value()) return absl::nullopt;
225   if (*ms == INT_MAX) return Duration::Infinity();
226   if (*ms == INT_MIN) return Duration::NegativeInfinity();
227   return Duration::Milliseconds(*ms);
228 }
229 
GetString(absl::string_view name) const230 absl::optional<absl::string_view> ChannelArgs::GetString(
231     absl::string_view name) const {
232   auto* v = Get(name);
233   if (v == nullptr) return absl::nullopt;
234   const auto* s = v->GetIfString();
235   if (s == nullptr) return absl::nullopt;
236   return *s;
237 }
238 
GetOwnedString(absl::string_view name) const239 absl::optional<std::string> ChannelArgs::GetOwnedString(
240     absl::string_view name) const {
241   absl::optional<absl::string_view> v = GetString(name);
242   if (!v.has_value()) return absl::nullopt;
243   return std::string(*v);
244 }
245 
GetVoidPointer(absl::string_view name) const246 void* ChannelArgs::GetVoidPointer(absl::string_view name) const {
247   auto* v = Get(name);
248   if (v == nullptr) return nullptr;
249   const auto* pp = v->GetIfPointer();
250   if (pp == nullptr) return nullptr;
251   return pp->c_pointer();
252 }
253 
GetBool(absl::string_view name) const254 absl::optional<bool> ChannelArgs::GetBool(absl::string_view name) const {
255   auto* v = Get(name);
256   if (v == nullptr) return absl::nullopt;
257   auto* i = v->GetIfInt();
258   if (i == nullptr) {
259     gpr_log(GPR_ERROR, "%s ignored: it must be an integer",
260             std::string(name).c_str());
261     return absl::nullopt;
262   }
263   switch (*i) {
264     case 0:
265       return false;
266     case 1:
267       return true;
268     default:
269       gpr_log(GPR_ERROR, "%s treated as bool but set to %d (assuming true)",
270               std::string(name).c_str(), *i);
271       return true;
272   }
273 }
274 
ToString() const275 std::string ChannelArgs::ToString() const {
276   std::vector<std::string> arg_strings;
277   args_.ForEach([&arg_strings](const std::string& key, const Value& value) {
278     std::string value_str;
279     if (auto* i = value.GetIfInt()) {
280       value_str = std::to_string(*i);
281     } else if (auto* s = value.GetIfString()) {
282       value_str = *s;
283     } else if (auto* p = value.GetIfPointer()) {
284       value_str = absl::StrFormat("%p", p->c_pointer());
285     }
286     arg_strings.push_back(absl::StrCat(key, "=", value_str));
287   });
288   return absl::StrCat("{", absl::StrJoin(arg_strings, ", "), "}");
289 }
290 
UnionWith(ChannelArgs other) const291 ChannelArgs ChannelArgs::UnionWith(ChannelArgs other) const {
292   if (args_.Empty()) return other;
293   if (other.args_.Empty()) return *this;
294   if (args_.Height() <= other.args_.Height()) {
295     args_.ForEach([&other](const std::string& key, const Value& value) {
296       other.args_ = other.args_.Add(key, value);
297     });
298     return other;
299   } else {
300     auto result = *this;
301     other.args_.ForEach([&result](const std::string& key, const Value& value) {
302       if (result.args_.Lookup(key) == nullptr) {
303         result.args_ = result.args_.Add(key, value);
304       }
305     });
306     return result;
307   }
308 }
309 
FuzzingReferenceUnionWith(ChannelArgs other) const310 ChannelArgs ChannelArgs::FuzzingReferenceUnionWith(ChannelArgs other) const {
311   // DO NOT OPTIMIZE THIS!!
312   args_.ForEach([&other](const std::string& key, const Value& value) {
313     other.args_ = other.args_.Add(key, value);
314   });
315   return other;
316 }
317 
operator ()(const grpc_channel_args * p) const318 void ChannelArgs::ChannelArgsDeleter::operator()(
319     const grpc_channel_args* p) const {
320   grpc_channel_args_destroy(p);
321 }
322 
operator <<(std::ostream & out,const ChannelArgs & args)323 std::ostream& operator<<(std::ostream& out, const ChannelArgs& args) {
324   return out << args.ToString();
325 }
326 
327 }  // namespace grpc_core
328 
copy_arg(const grpc_arg * src)329 static grpc_arg copy_arg(const grpc_arg* src) {
330   grpc_arg dst;
331   dst.type = src->type;
332   dst.key = gpr_strdup(src->key);
333   switch (dst.type) {
334     case GRPC_ARG_STRING:
335       dst.value.string = gpr_strdup(src->value.string);
336       break;
337     case GRPC_ARG_INTEGER:
338       dst.value.integer = src->value.integer;
339       break;
340     case GRPC_ARG_POINTER:
341       dst.value.pointer = src->value.pointer;
342       dst.value.pointer.p =
343           src->value.pointer.vtable->copy(src->value.pointer.p);
344       break;
345   }
346   return dst;
347 }
348 
grpc_channel_args_copy_and_add(const grpc_channel_args * src,const grpc_arg * to_add,size_t num_to_add)349 grpc_channel_args* grpc_channel_args_copy_and_add(const grpc_channel_args* src,
350                                                   const grpc_arg* to_add,
351                                                   size_t num_to_add) {
352   return grpc_channel_args_copy_and_add_and_remove(src, nullptr, 0, to_add,
353                                                    num_to_add);
354 }
355 
grpc_channel_args_copy_and_remove(const grpc_channel_args * src,const char ** to_remove,size_t num_to_remove)356 grpc_channel_args* grpc_channel_args_copy_and_remove(
357     const grpc_channel_args* src, const char** to_remove,
358     size_t num_to_remove) {
359   return grpc_channel_args_copy_and_add_and_remove(src, to_remove,
360                                                    num_to_remove, nullptr, 0);
361 }
362 
should_remove_arg(const grpc_arg * arg,const char ** to_remove,size_t num_to_remove)363 static bool should_remove_arg(const grpc_arg* arg, const char** to_remove,
364                               size_t num_to_remove) {
365   for (size_t i = 0; i < num_to_remove; ++i) {
366     if (strcmp(arg->key, to_remove[i]) == 0) return true;
367   }
368   return false;
369 }
370 
grpc_channel_args_copy_and_add_and_remove(const grpc_channel_args * src,const char ** to_remove,size_t num_to_remove,const grpc_arg * to_add,size_t num_to_add)371 grpc_channel_args* grpc_channel_args_copy_and_add_and_remove(
372     const grpc_channel_args* src, const char** to_remove, size_t num_to_remove,
373     const grpc_arg* to_add, size_t num_to_add) {
374   // Figure out how many args we'll be copying.
375   size_t num_args_to_copy = 0;
376   if (src != nullptr) {
377     for (size_t i = 0; i < src->num_args; ++i) {
378       if (!should_remove_arg(&src->args[i], to_remove, num_to_remove)) {
379         ++num_args_to_copy;
380       }
381     }
382   }
383   // Create result.
384   grpc_channel_args* dst =
385       static_cast<grpc_channel_args*>(gpr_malloc(sizeof(grpc_channel_args)));
386   dst->num_args = num_args_to_copy + num_to_add;
387   if (dst->num_args == 0) {
388     dst->args = nullptr;
389     return dst;
390   }
391   dst->args =
392       static_cast<grpc_arg*>(gpr_malloc(sizeof(grpc_arg) * dst->num_args));
393   // Copy args from src that are not being removed.
394   size_t dst_idx = 0;
395   if (src != nullptr) {
396     for (size_t i = 0; i < src->num_args; ++i) {
397       if (!should_remove_arg(&src->args[i], to_remove, num_to_remove)) {
398         dst->args[dst_idx++] = copy_arg(&src->args[i]);
399       }
400     }
401   }
402   // Add args from to_add.
403   for (size_t i = 0; i < num_to_add; ++i) {
404     dst->args[dst_idx++] = copy_arg(&to_add[i]);
405   }
406   GPR_ASSERT(dst_idx == dst->num_args);
407   return dst;
408 }
409 
grpc_channel_args_copy(const grpc_channel_args * src)410 grpc_channel_args* grpc_channel_args_copy(const grpc_channel_args* src) {
411   return grpc_channel_args_copy_and_add(src, nullptr, 0);
412 }
413 
grpc_channel_args_union(const grpc_channel_args * a,const grpc_channel_args * b)414 grpc_channel_args* grpc_channel_args_union(const grpc_channel_args* a,
415                                            const grpc_channel_args* b) {
416   if (a == nullptr) return grpc_channel_args_copy(b);
417   if (b == nullptr) return grpc_channel_args_copy(a);
418   const size_t max_out = (a->num_args + b->num_args);
419   grpc_arg* uniques =
420       static_cast<grpc_arg*>(gpr_malloc(sizeof(*uniques) * max_out));
421   for (size_t i = 0; i < a->num_args; ++i) uniques[i] = a->args[i];
422 
423   size_t uniques_idx = a->num_args;
424   for (size_t i = 0; i < b->num_args; ++i) {
425     const char* b_key = b->args[i].key;
426     if (grpc_channel_args_find(a, b_key) == nullptr) {  // not found
427       uniques[uniques_idx++] = b->args[i];
428     }
429   }
430   grpc_channel_args* result =
431       grpc_channel_args_copy_and_add(nullptr, uniques, uniques_idx);
432   gpr_free(uniques);
433   return result;
434 }
435 
cmp_arg(const grpc_arg * a,const grpc_arg * b)436 static int cmp_arg(const grpc_arg* a, const grpc_arg* b) {
437   int c = grpc_core::QsortCompare(a->type, b->type);
438   if (c != 0) return c;
439   c = strcmp(a->key, b->key);
440   if (c != 0) return c;
441   switch (a->type) {
442     case GRPC_ARG_STRING:
443       return strcmp(a->value.string, b->value.string);
444     case GRPC_ARG_INTEGER:
445       return grpc_core::QsortCompare(a->value.integer, b->value.integer);
446     case GRPC_ARG_POINTER:
447       return grpc_core::channel_args_detail::PointerCompare(
448           a->value.pointer.p, a->value.pointer.vtable, b->value.pointer.p,
449           b->value.pointer.vtable);
450   }
451   GPR_UNREACHABLE_CODE(return 0);
452 }
453 
454 // stabilizing comparison function: since channel_args ordering matters for
455 // keys with the same name, we need to preserve that ordering
cmp_key_stable(const void * ap,const void * bp)456 static int cmp_key_stable(const void* ap, const void* bp) {
457   const grpc_arg* const* a = static_cast<const grpc_arg* const*>(ap);
458   const grpc_arg* const* b = static_cast<const grpc_arg* const*>(bp);
459   int c = strcmp((*a)->key, (*b)->key);
460   if (c == 0) c = grpc_core::QsortCompare(*a, *b);
461   return c;
462 }
463 
grpc_channel_args_normalize(const grpc_channel_args * src)464 grpc_channel_args* grpc_channel_args_normalize(const grpc_channel_args* src) {
465   grpc_arg** args =
466       static_cast<grpc_arg**>(gpr_malloc(sizeof(grpc_arg*) * src->num_args));
467   for (size_t i = 0; i < src->num_args; i++) {
468     args[i] = &src->args[i];
469   }
470   if (src->num_args > 1) {
471     qsort(args, src->num_args, sizeof(grpc_arg*), cmp_key_stable);
472   }
473 
474   grpc_channel_args* b =
475       static_cast<grpc_channel_args*>(gpr_malloc(sizeof(grpc_channel_args)));
476   b->num_args = src->num_args;
477   b->args = static_cast<grpc_arg*>(gpr_malloc(sizeof(grpc_arg) * b->num_args));
478   for (size_t i = 0; i < src->num_args; i++) {
479     b->args[i] = copy_arg(args[i]);
480   }
481 
482   gpr_free(args);
483   return b;
484 }
485 
grpc_channel_args_destroy(grpc_channel_args * a)486 void grpc_channel_args_destroy(grpc_channel_args* a) {
487   size_t i;
488   if (!a) return;
489   for (i = 0; i < a->num_args; i++) {
490     switch (a->args[i].type) {
491       case GRPC_ARG_STRING:
492         gpr_free(a->args[i].value.string);
493         break;
494       case GRPC_ARG_INTEGER:
495         break;
496       case GRPC_ARG_POINTER:
497         a->args[i].value.pointer.vtable->destroy(a->args[i].value.pointer.p);
498         break;
499     }
500     gpr_free(a->args[i].key);
501   }
502   gpr_free(a->args);
503   gpr_free(a);
504 }
505 
grpc_channel_args_compare(const grpc_channel_args * a,const grpc_channel_args * b)506 int grpc_channel_args_compare(const grpc_channel_args* a,
507                               const grpc_channel_args* b) {
508   if (a == nullptr && b == nullptr) return 0;
509   if (a == nullptr || b == nullptr) return a == nullptr ? -1 : 1;
510   int c = grpc_core::QsortCompare(a->num_args, b->num_args);
511   if (c != 0) return c;
512   for (size_t i = 0; i < a->num_args; i++) {
513     c = cmp_arg(&a->args[i], &b->args[i]);
514     if (c != 0) return c;
515   }
516   return 0;
517 }
518 
grpc_channel_args_find(const grpc_channel_args * args,const char * name)519 const grpc_arg* grpc_channel_args_find(const grpc_channel_args* args,
520                                        const char* name) {
521   if (args != nullptr) {
522     for (size_t i = 0; i < args->num_args; ++i) {
523       if (strcmp(args->args[i].key, name) == 0) {
524         return &args->args[i];
525       }
526     }
527   }
528   return nullptr;
529 }
530 
grpc_channel_arg_get_integer(const grpc_arg * arg,const grpc_integer_options options)531 int grpc_channel_arg_get_integer(const grpc_arg* arg,
532                                  const grpc_integer_options options) {
533   if (arg == nullptr) return options.default_value;
534   if (arg->type != GRPC_ARG_INTEGER) {
535     gpr_log(GPR_ERROR, "%s ignored: it must be an integer", arg->key);
536     return options.default_value;
537   }
538   if (arg->value.integer < options.min_value) {
539     gpr_log(GPR_ERROR, "%s ignored: it must be >= %d", arg->key,
540             options.min_value);
541     return options.default_value;
542   }
543   if (arg->value.integer > options.max_value) {
544     gpr_log(GPR_ERROR, "%s ignored: it must be <= %d", arg->key,
545             options.max_value);
546     return options.default_value;
547   }
548   return arg->value.integer;
549 }
550 
grpc_channel_args_find_integer(const grpc_channel_args * args,const char * name,const grpc_integer_options options)551 int grpc_channel_args_find_integer(const grpc_channel_args* args,
552                                    const char* name,
553                                    const grpc_integer_options options) {
554   const grpc_arg* arg = grpc_channel_args_find(args, name);
555   return grpc_channel_arg_get_integer(arg, options);
556 }
557 
grpc_channel_arg_get_string(const grpc_arg * arg)558 char* grpc_channel_arg_get_string(const grpc_arg* arg) {
559   if (arg == nullptr) return nullptr;
560   if (arg->type != GRPC_ARG_STRING) {
561     gpr_log(GPR_ERROR, "%s ignored: it must be an string", arg->key);
562     return nullptr;
563   }
564   return arg->value.string;
565 }
566 
grpc_channel_args_find_string(const grpc_channel_args * args,const char * name)567 char* grpc_channel_args_find_string(const grpc_channel_args* args,
568                                     const char* name) {
569   const grpc_arg* arg = grpc_channel_args_find(args, name);
570   return grpc_channel_arg_get_string(arg);
571 }
572 
grpc_channel_arg_get_bool(const grpc_arg * arg,bool default_value)573 bool grpc_channel_arg_get_bool(const grpc_arg* arg, bool default_value) {
574   if (arg == nullptr) return default_value;
575   if (arg->type != GRPC_ARG_INTEGER) {
576     gpr_log(GPR_ERROR, "%s ignored: it must be an integer", arg->key);
577     return default_value;
578   }
579   switch (arg->value.integer) {
580     case 0:
581       return false;
582     case 1:
583       return true;
584     default:
585       gpr_log(GPR_ERROR, "%s treated as bool but set to %d (assuming true)",
586               arg->key, arg->value.integer);
587       return true;
588   }
589 }
590 
grpc_channel_args_find_bool(const grpc_channel_args * args,const char * name,bool default_value)591 bool grpc_channel_args_find_bool(const grpc_channel_args* args,
592                                  const char* name, bool default_value) {
593   const grpc_arg* arg = grpc_channel_args_find(args, name);
594   return grpc_channel_arg_get_bool(arg, default_value);
595 }
596 
grpc_channel_args_want_minimal_stack(const grpc_channel_args * args)597 bool grpc_channel_args_want_minimal_stack(const grpc_channel_args* args) {
598   return grpc_channel_arg_get_bool(
599       grpc_channel_args_find(args, GRPC_ARG_MINIMAL_STACK), false);
600 }
601 
grpc_channel_arg_string_create(char * name,char * value)602 grpc_arg grpc_channel_arg_string_create(char* name, char* value) {
603   grpc_arg arg;
604   arg.type = GRPC_ARG_STRING;
605   arg.key = name;
606   arg.value.string = value;
607   return arg;
608 }
609 
grpc_channel_arg_integer_create(char * name,int value)610 grpc_arg grpc_channel_arg_integer_create(char* name, int value) {
611   grpc_arg arg;
612   arg.type = GRPC_ARG_INTEGER;
613   arg.key = name;
614   arg.value.integer = value;
615   return arg;
616 }
617 
grpc_channel_arg_pointer_create(char * name,void * value,const grpc_arg_pointer_vtable * vtable)618 grpc_arg grpc_channel_arg_pointer_create(
619     char* name, void* value, const grpc_arg_pointer_vtable* vtable) {
620   grpc_arg arg;
621   arg.type = GRPC_ARG_POINTER;
622   arg.key = name;
623   arg.value.pointer.p = value;
624   arg.value.pointer.vtable = vtable;
625   return arg;
626 }
627 
grpc_channel_args_string(const grpc_channel_args * args)628 std::string grpc_channel_args_string(const grpc_channel_args* args) {
629   return grpc_core::ChannelArgs::FromC(args).ToString();
630 }
631 
632 namespace grpc_core {
ChannelArgsBuiltinPrecondition(const grpc_channel_args * src)633 ChannelArgs ChannelArgsBuiltinPrecondition(const grpc_channel_args* src) {
634   if (src == nullptr) return ChannelArgs();
635   ChannelArgs output;
636   std::map<absl::string_view, std::vector<absl::string_view>>
637       concatenated_values;
638   for (size_t i = 0; i < src->num_args; i++) {
639     absl::string_view key = src->args[i].key;
640     // User-agent strings were traditionally multi-valued and concatenated.
641     // We preserve this behavior for backwards compatibility.
642     if (key == GRPC_ARG_PRIMARY_USER_AGENT_STRING ||
643         key == GRPC_ARG_SECONDARY_USER_AGENT_STRING) {
644       if (src->args[i].type != GRPC_ARG_STRING) {
645         gpr_log(GPR_ERROR, "Channel argument '%s' should be a string",
646                 std::string(key).c_str());
647       } else {
648         concatenated_values[key].push_back(src->args[i].value.string);
649       }
650       continue;
651     } else if (absl::StartsWith(key, "grpc.internal.")) {
652       continue;
653     }
654     if (!output.Contains(key)) {
655       output = output.Set(src->args[i]);
656     } else {
657       // Traditional grpc_channel_args_find behavior was to pick the first
658       // value.
659       // For compatibility with existing users, we will do the same here.
660     }
661   }
662   // Concatenate the concatenated values.
663   for (const auto& concatenated_value : concatenated_values) {
664     output = output.Set(concatenated_value.first,
665                         absl::StrJoin(concatenated_value.second, " "));
666   }
667   return output;
668 }
669 
670 }  // namespace grpc_core
671 
672 namespace {
673 grpc_channel_args_client_channel_creation_mutator g_mutator = nullptr;
674 }  // namespace
675 
grpc_channel_args_set_client_channel_creation_mutator(grpc_channel_args_client_channel_creation_mutator cb)676 void grpc_channel_args_set_client_channel_creation_mutator(
677     grpc_channel_args_client_channel_creation_mutator cb) {
678   GPR_DEBUG_ASSERT(g_mutator == nullptr);
679   g_mutator = cb;
680 }
681 grpc_channel_args_client_channel_creation_mutator
grpc_channel_args_get_client_channel_creation_mutator()682 grpc_channel_args_get_client_channel_creation_mutator() {
683   return g_mutator;
684 }
685