1cac7dca0Sopenharmony_ci// Copyright (c) 2023 Huawei Device Co., Ltd.
2cac7dca0Sopenharmony_ci// Licensed under the Apache License, Version 2.0 (the "License");
3cac7dca0Sopenharmony_ci// you may not use this file except in compliance with the License.
4cac7dca0Sopenharmony_ci// You may obtain a copy of the License at
5cac7dca0Sopenharmony_ci//
6cac7dca0Sopenharmony_ci//     http://www.apache.org/licenses/LICENSE-2.0
7cac7dca0Sopenharmony_ci//
8cac7dca0Sopenharmony_ci// Unless required by applicable law or agreed to in writing, software
9cac7dca0Sopenharmony_ci// distributed under the License is distributed on an "AS IS" BASIS,
10cac7dca0Sopenharmony_ci// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
11cac7dca0Sopenharmony_ci// See the License for the specific language governing permissions and
12cac7dca0Sopenharmony_ci// limitations under the License.
13cac7dca0Sopenharmony_ci
14cac7dca0Sopenharmony_ci//! Macros for use with ylong_runtime
15cac7dca0Sopenharmony_ci
16cac7dca0Sopenharmony_ci#![allow(clippy::needless_doctest_main)]
17cac7dca0Sopenharmony_ci#![doc(test(no_crate_inject,))]
18cac7dca0Sopenharmony_ci
19cac7dca0Sopenharmony_cimod select;
20cac7dca0Sopenharmony_ci
21cac7dca0Sopenharmony_ciuse proc_macro::{Delimiter, Group, Punct, Spacing, TokenStream, TokenTree};
22cac7dca0Sopenharmony_ci
23cac7dca0Sopenharmony_ci/// Implementation detail of the `select!` macro. This macro is **not** intended
24cac7dca0Sopenharmony_ci/// to be used as part of the public API.
25cac7dca0Sopenharmony_ci/// # Examples
26cac7dca0Sopenharmony_ci///
27cac7dca0Sopenharmony_ci/// ```
28cac7dca0Sopenharmony_ci/// #[derive(PartialEq, Debug)]
29cac7dca0Sopenharmony_ci/// enum Out {
30cac7dca0Sopenharmony_ci///     Finish,
31cac7dca0Sopenharmony_ci///     Fail,
32cac7dca0Sopenharmony_ci/// }
33cac7dca0Sopenharmony_ci/// let tuple = ylong_runtime_macros::tuple_form!(( (((0)+1)+1) ) with Out::Fail except Out::Finish at ( ) );
34cac7dca0Sopenharmony_ci/// assert_eq!(tuple, (Out::Finish, Out::Fail));
35cac7dca0Sopenharmony_ci/// ```
36cac7dca0Sopenharmony_ci#[proc_macro]
37cac7dca0Sopenharmony_ci#[doc(hidden)]
38cac7dca0Sopenharmony_cipub fn tuple_form(input: TokenStream) -> TokenStream {
39cac7dca0Sopenharmony_ci    let tuple_parser = select::tuple_parser(input);
40cac7dca0Sopenharmony_ci
41cac7dca0Sopenharmony_ci    let mut group_inner = TokenStream::new();
42cac7dca0Sopenharmony_ci
43cac7dca0Sopenharmony_ci    // Constructing Tuples
44cac7dca0Sopenharmony_ci    for i in 0..tuple_parser.len {
45cac7dca0Sopenharmony_ci        if i == tuple_parser.except_index {
46cac7dca0Sopenharmony_ci            // Set 'except_index' at index
47cac7dca0Sopenharmony_ci            group_inner.extend(tuple_parser.except.clone());
48cac7dca0Sopenharmony_ci        } else {
49cac7dca0Sopenharmony_ci            // Set 'default'
50cac7dca0Sopenharmony_ci            group_inner.extend(tuple_parser.default.clone());
51cac7dca0Sopenharmony_ci        }
52cac7dca0Sopenharmony_ci        // Add ',' separator
53cac7dca0Sopenharmony_ci        if i != tuple_parser.len - 1 {
54cac7dca0Sopenharmony_ci            let punct: Punct = Punct::new(',', Spacing::Alone);
55cac7dca0Sopenharmony_ci            group_inner.extend(TokenStream::from(TokenTree::from(punct)));
56cac7dca0Sopenharmony_ci        }
57cac7dca0Sopenharmony_ci    }
58cac7dca0Sopenharmony_ci    // Add parentheses on the outermost
59cac7dca0Sopenharmony_ci    let tuple = Group::new(Delimiter::Parenthesis, group_inner);
60cac7dca0Sopenharmony_ci
61cac7dca0Sopenharmony_ci    TokenStream::from(TokenTree::from(tuple))
62cac7dca0Sopenharmony_ci}
63