| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148 |
- // unsafe.go is based on qtls/unsafe.go
- // https://github.com/quic-go/qtls-go1-20/blob/49f389c17d984e5b248f0a57cbae10dd4198a3bf/unsafe.go
- /* Copyright (c) 2009 The Go Authors. All rights reserved.
- Redistribution and use in source and binary forms, with or without
- modification, are permitted provided that the following conditions are
- met:
- * Redistributions of source code must retain the above copyright
- notice, this list of conditions and the following disclaimer.
- * Redistributions in binary form must reproduce the above
- copyright notice, this list of conditions and the following disclaimer
- in the documentation and/or other materials provided with the
- distribution.
- * Neither the name of Google Inc. nor the names of its
- contributors may be used to endorse or promote products derived from
- this software without specific prior written permission.
- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
- "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
- LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
- A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
- OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
- SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
- LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
- DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
- THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
- OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- */
- package tls
- import (
- "crypto/tls"
- "fmt"
- "reflect"
- "unsafe"
- )
- func init() {
- err := structsEqual(&tls.ConnectionState{}, &ConnectionState{})
- if err != nil {
- panic(fmt.Sprintf("tls: ConnectionState is not equal to tls.ConnectionState: %v", err))
- }
- }
- func UnsafeFromConnectionState(ss *ConnectionState) *tls.ConnectionState {
- return (*tls.ConnectionState)(unsafe.Pointer(ss))
- }
- func UnsafeToConnectionState(ss *tls.ConnectionState) *ConnectionState {
- return (*ConnectionState)(unsafe.Pointer(ss))
- }
- func structsEqual(a, b interface{}) error {
- return compare(reflect.TypeOf(a), reflect.TypeOf(b))
- }
- // compare compares two types and returns true if and only if
- // they can be casted to each other safely.
- // compare does not currently support Maps, Chan, UnsafePointer if reflect.DeepEqual fails.
- // Support for these types can be added if needed.
- // note that field names are still compared.
- func compare(a, b reflect.Type) error {
- if reflect.DeepEqual(a, b) {
- return nil
- }
- if isPrimitive(a) && isPrimitive(b) && a.Kind() == b.Kind() {
- return nil
- }
- if a.Kind() != b.Kind() {
- return fmt.Errorf("kind mismatch: %s vs %s", a.Kind().String(), b.Kind().String())
- }
- if a.Kind() == reflect.Pointer || a.Kind() == reflect.Slice {
- return compare(a.Elem(), b.Elem())
- }
- if a.Kind() == reflect.Func {
- if a.NumIn() != b.NumIn() || a.NumOut() != b.NumOut() {
- return fmt.Errorf(
- "function signature mismatch: different number of input or output parameters: NumIn: %d vs %d, NumOut: %d vs %d",
- a.NumIn(), b.NumIn(), a.NumOut(), b.NumOut(),
- )
- }
- for i_in := 0; i_in < a.NumIn(); i_in++ {
- err := compare(a.In(i_in), b.In(i_in))
- if err != nil {
- return fmt.Errorf("function %s input parameter mismatch at index %d: %v", a.Name(), i_in, err)
- }
- }
- for i_out := 0; i_out < a.NumOut(); i_out++ {
- err := compare(a.Out(i_out), b.Out(i_out))
- if err != nil {
- return fmt.Errorf("function %s output parameter mismatch at index %d: %v", a.Name(), i_out, err)
- }
- }
- return nil
- }
- if a.Kind() == reflect.Struct {
- if a.NumField() != b.NumField() {
- return fmt.Errorf("struct field count mismatch: %d vs %d", a.NumField(), b.NumField())
- }
- for i := 0; i < a.NumField(); i++ {
- fa := a.Field(i)
- fb := b.Field(i)
- if !reflect.DeepEqual(fa.Index, fb.Index) || fa.Name != fb.Name ||
- fa.Anonymous != fb.Anonymous || fa.Offset != fb.Offset {
- return fmt.Errorf("struct field mismatch at index %d: %+v vs %+v", i, fa, fb)
- }
- if !reflect.DeepEqual(fa.Type, fb.Type) {
- err := compare(fa.Type, fb.Type)
- if err != nil {
- return fmt.Errorf("struct %s field type mismatch at index %d with name %s: %v", a.Name(), i, fa.Name, err)
- }
- }
- }
- return nil
- }
- // TODO: add support for missing types
- return fmt.Errorf("unsupported type: %s for field %s", a.Kind().String(), a.Name())
- }
- // isPrimitive checks if a reflect.Type represents a primitive type
- func isPrimitive(t reflect.Type) bool {
- switch t.Kind() {
- case reflect.Bool,
- reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
- reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64,
- reflect.Float32, reflect.Float64,
- reflect.Complex64, reflect.Complex128,
- reflect.String:
- return true
- default:
- return false
- }
- }
|