13af6ab5fSopenharmony_ci/*
23af6ab5fSopenharmony_ci * Copyright (c) 2023-2024 Huawei Device Co., Ltd.
33af6ab5fSopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
43af6ab5fSopenharmony_ci * you may not use this file except in compliance with the License.
53af6ab5fSopenharmony_ci * You may obtain a copy of the License at
63af6ab5fSopenharmony_ci *
73af6ab5fSopenharmony_ci * http://www.apache.org/licenses/LICENSE-2.0
83af6ab5fSopenharmony_ci *
93af6ab5fSopenharmony_ci * Unless required by applicable law or agreed to in writing, software
103af6ab5fSopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
113af6ab5fSopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
123af6ab5fSopenharmony_ci * See the License for the specific language governing permissions and
133af6ab5fSopenharmony_ci * limitations under the License.
143af6ab5fSopenharmony_ci */
153af6ab5fSopenharmony_ci
163af6ab5fSopenharmony_cifunction assert_ccexc(f: () => void) {
173af6ab5fSopenharmony_ci    try {
183af6ab5fSopenharmony_ci        f();
193af6ab5fSopenharmony_ci    } catch (e) {
203af6ab5fSopenharmony_ci        assert(e instanceof ClassCastError);
213af6ab5fSopenharmony_ci        return;
223af6ab5fSopenharmony_ci    }
233af6ab5fSopenharmony_ci    assert false : "exception expected";
243af6ab5fSopenharmony_ci}
253af6ab5fSopenharmony_ci
263af6ab5fSopenharmony_ciclass A { }
273af6ab5fSopenharmony_ciclass B { }
283af6ab5fSopenharmony_ciclass C { }
293af6ab5fSopenharmony_ciclass X<T> { }
303af6ab5fSopenharmony_ci
313af6ab5fSopenharmony_cifunction erase<T>(x: Object | null | undefined): T { return x as T; }
323af6ab5fSopenharmony_ci
333af6ab5fSopenharmony_cifunction test_substitution() {
343af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<Object>(null); })
353af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<Object>(undefined); })
363af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<A>(null); })
373af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<A>(undefined); })
383af6ab5fSopenharmony_ci
393af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<Object | undefined>(null); })
403af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<Object | null>(undefined); })
413af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<A | undefined>(null); })
423af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<A | null>(undefined); })
433af6ab5fSopenharmony_ci
443af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<A>(undefined); })
453af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<A>(new Object()); })
463af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<A>(new B()); })
473af6ab5fSopenharmony_ci
483af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<A | B>(undefined); })
493af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<A | B>(new Object()); })
503af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<A | B>(new C()); })
513af6ab5fSopenharmony_ci
523af6ab5fSopenharmony_ci    assert_ccexc(() => { erase<A[]>(new B[0]); })
533af6ab5fSopenharmony_ci}
543af6ab5fSopenharmony_ci
553af6ab5fSopenharmony_ciclass Erased<T> {
563af6ab5fSopenharmony_ci    constructor(x: Object | null | undefined) { this.t = x as T; }
573af6ab5fSopenharmony_ci    t: T;
583af6ab5fSopenharmony_ci}
593af6ab5fSopenharmony_ci
603af6ab5fSopenharmony_cifunction test_substitution_memberexpr() {
613af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<Object>(null).t; })
623af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<Object>(undefined).t; })
633af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<A>(null).t; })
643af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<A>(undefined).t; })
653af6ab5fSopenharmony_ci
663af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<Object | undefined>(null).t; })
673af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<Object | null>(undefined).t; })
683af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<A | undefined>(null).t; })
693af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<A | null>(undefined).t; })
703af6ab5fSopenharmony_ci
713af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<A>(undefined).t; })
723af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<A>(new Object()).t; })
733af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<A>(new B()).t; })
743af6ab5fSopenharmony_ci
753af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<A | B>(undefined).t; })
763af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<A | B>(new Object()).t; })
773af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<A | B>(new C()).t; })
783af6ab5fSopenharmony_ci
793af6ab5fSopenharmony_ci    assert_ccexc(() => { new Erased<A[]>(new B[0]).t; })
803af6ab5fSopenharmony_ci}
813af6ab5fSopenharmony_ci
823af6ab5fSopenharmony_cifunction cast_to_tparam<T extends A | B | null>(x: Object | null | undefined) { x as T; }
833af6ab5fSopenharmony_ci
843af6ab5fSopenharmony_cifunction test_constraint() {
853af6ab5fSopenharmony_ci    assert_ccexc(() => { cast_to_tparam<A>(undefined); })
863af6ab5fSopenharmony_ci    assert_ccexc(() => { cast_to_tparam<A>(new Object()); })
873af6ab5fSopenharmony_ci    assert_ccexc(() => { cast_to_tparam<A>(new C()); })
883af6ab5fSopenharmony_ci}
893af6ab5fSopenharmony_ci
903af6ab5fSopenharmony_cifunction to_basetype<T>(x: Object | null | undefined) { return x as X<T>; }
913af6ab5fSopenharmony_ci
923af6ab5fSopenharmony_cifunction test_basetype() {
933af6ab5fSopenharmony_ci    assert_ccexc(() => { to_basetype<A>(null); })
943af6ab5fSopenharmony_ci    assert_ccexc(() => { to_basetype<A>(undefined); })
953af6ab5fSopenharmony_ci    assert_ccexc(() => { to_basetype<A>(new Object()); })
963af6ab5fSopenharmony_ci    assert_ccexc(() => { to_basetype<A>(new C()); })
973af6ab5fSopenharmony_ci}
983af6ab5fSopenharmony_ci
993af6ab5fSopenharmony_cifunction main() {
1003af6ab5fSopenharmony_ci    test_substitution();
1013af6ab5fSopenharmony_ci    test_substitution_memberexpr();
1023af6ab5fSopenharmony_ci    test_constraint();
1033af6ab5fSopenharmony_ci    test_basetype();
1043af6ab5fSopenharmony_ci}
105