| /* |
| * Licensed to the Apache Software Foundation (ASF) under one |
| * or more contributor license agreements. See the NOTICE file |
| * distributed with this work for additional information |
| * regarding copyright ownership. The ASF licenses this file |
| * to you under the Apache License, Version 2.0 (the |
| * "License"); you may not use this file except in compliance |
| * with the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, |
| * software distributed under the License is distributed on an |
| * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
| * KIND, either express or implied. See the License for the |
| * specific language governing permissions and limitations |
| * under the License. |
| */ |
| module thrift.internal.traits; |
| |
| import std.traits; |
| |
| /** |
| * Adds or removes attributes from the given function (pointer) or delegate |
| * type. |
| * |
| * Besides the base type, two std.traits.FunctionAttribute bitfields are |
| * accepted, representing the attributes to add to the function signature |
| * resp. to remove from it. If an attribute appears in both fields, an error |
| * is raised. |
| * |
| * Params: |
| * T = The base type. |
| * setAttrs = The attributes to add to the type, if not already present. |
| * clearAttrs = The attributes to remove from the type, if present. |
| */ |
| template ChangeFuncAttrs( |
| T, |
| FunctionAttribute setAttrs = FunctionAttribute.none, |
| FunctionAttribute clearAttrs = FunctionAttribute.none |
| ) if (isFunctionPointer!T || isDelegate!T) { |
| static assert(!(setAttrs & clearAttrs), |
| "Cannot set and clear attributes at the same time."); |
| mixin({ |
| enum newAttrs = (functionAttributes!T | setAttrs) & ~clearAttrs; |
| static assert(!(newAttrs & FunctionAttribute.trusted) || |
| !(newAttrs & FunctionAttribute.safe), |
| "Cannot have a function/delegate that is both trusted and safe."); |
| |
| string result = "alias "; |
| |
| static if (functionLinkage!T != "D") { |
| result ~= "extern(" ~ functionLinkage!T ~ ") "; |
| } |
| |
| static if (newAttrs & FunctionAttribute.ref_) { |
| result ~= "ref "; |
| } |
| |
| result ~= "ReturnType!T"; |
| |
| static if (isDelegate!T) { |
| result ~= " delegate"; |
| } else { |
| result ~= " function"; |
| } |
| |
| result ~= "(ParameterTypeTuple!T)"; |
| |
| static if (newAttrs & FunctionAttribute.pure_) { |
| result ~= " pure"; |
| } |
| static if (newAttrs & FunctionAttribute.nothrow_) { |
| result ~= " nothrow"; |
| } |
| static if (newAttrs & FunctionAttribute.property) { |
| result ~= " @property"; |
| } |
| static if (newAttrs & FunctionAttribute.trusted) { |
| result ~= " @trusted"; |
| } |
| static if (newAttrs & FunctionAttribute.safe) { |
| result ~= " @safe"; |
| } |
| |
| result ~= " ChangeFuncAttrs;"; |
| return result; |
| }()); |
| } |
| |
| /// Ditto |
| template ChangeFuncAttrs( |
| T, |
| FunctionAttribute setAttrs = FunctionAttribute.none, |
| FunctionAttribute clearAttrs = FunctionAttribute.none |
| ) if (is(T == function)) { |
| // To avoid a lot of syntactic headaches, we just use the above version to |
| // operate on the corresponding function pointer type and then remove the |
| // pointer again. |
| alias FunctionTypeOf!(ChangeFuncAttrs!(T*, setAttrs, clearAttrs)) |
| ChangeFuncAttrs; |
| } |
| |
| version (unittest) { |
| import std.algorithm; |
| import std.metastrings; |
| import std.typetuple; |
| } |
| unittest { |
| alias FunctionAttribute FA; |
| foreach (T0; TypeTuple!( |
| int function(int), |
| int delegate(int), |
| FunctionTypeOf!(int function(int)) |
| )) { |
| alias ChangeFuncAttrs!(T0, FA.safe) T1; |
| static assert(functionAttributes!T1 == FA.safe); |
| |
| alias ChangeFuncAttrs!(T1, FA.nothrow_ | FA.ref_, FA.safe) T2; |
| static assert(functionAttributes!T2 == (FA.nothrow_ | FA.ref_)); |
| |
| enum allAttrs = reduce!"a | b"([EnumMembers!FA]) & ~FA.safe; |
| |
| alias ChangeFuncAttrs!(T2, allAttrs) T3; |
| static assert(functionAttributes!T3 == allAttrs); |
| |
| alias ChangeFuncAttrs!(T3, FA.none, allAttrs) T4; |
| static assert(is(T4 == T0)); |
| } |
| } |
| |
| /** |
| * Adds »nothrow« to the type of the passed function pointer/delegate, if it |
| * is not already present. |
| * |
| * Technically, assumeNothrow just performs a cast, but using it has the |
| * advantage of being explicitly about the operation that is performed. |
| */ |
| auto assumeNothrow(T)(T t) if (isFunctionPointer!T || isDelegate!T) { |
| return cast(ChangeFuncAttrs!(T, FunctionAttribute.nothrow_))t; |
| } |