using System.Numerics; using System.Runtime.InteropServices; using System.Security; namespace Raylib_cs; // NOTE: Helper types to be used instead of array return types for *ToFloat functions public unsafe struct Float3 { public fixed float v[3]; } public unsafe struct Float16 { public fixed float v[16]; } [SuppressUnmanagedCodeSecurity] public static unsafe partial class Raymath { /// /// Used by DllImport to load the native library /// public const string NativeLibName = "raylib"; /// Clamp float value [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Clamp(float value, float min, float max); /// Calculate linear interpolation between two vectors [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Lerp(float start, float end, float amount); /// Normalize input value within input range [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Normalize(float value, float start, float end); /// Remap input value within input range to output range [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Remap( float value, float inputStart, float inputEnd, float outputStart, float outputEnd ); /// Wrap input value from min to max [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Wrap(float value, float min, float max); /// Check whether two given floats are almost equal [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern int FloatEquals(float x, float y); /// Vector with components value 0.0f [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Zero(); /// Vector with components value 1.0f [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2One(); /// Add two vectors (v1 + v2) [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Add(Vector2 v1, Vector2 v2); /// Add vector and float value [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2AddValue(Vector2 v, float add); /// Subtract two vectors (v1 - v2) [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Subtract(Vector2 v1, Vector2 v2); /// Subtract vector by float value [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2SubtractValue(Vector2 v, float sub); /// Calculate vector length [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Vector2Length(Vector2 v); /// Calculate vector square length [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Vector2LengthSqr(Vector2 v); /// Calculate two vectors dot product [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Vector2DotProduct(Vector2 v1, Vector2 v2); /// Calculate distance between two vectors [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Vector2Distance(Vector2 v1, Vector2 v2); /// Calculate square distance between two vectors [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Vector2DistanceSqr(Vector2 v1, Vector2 v2); /// /// Calculate angle between two vectors /// NOTE: Angle is calculated from origin point (0, 0) /// [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Vector2Angle(Vector2 v1, Vector2 v2); /// /// Calculate angle defined by a two vectors line /// NOTE: Parameters need to be normalized /// Current implementation should be aligned with glm::angle /// [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Vector2LineAngle(Vector2 start, Vector2 end); /// Scale vector (multiply by value) [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Scale(Vector2 v, float scale); /// Multiply vector by vector [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Multiply(Vector2 v1, Vector2 v2); /// Negate vector [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Negate(Vector2 v); /// Divide vector by vector [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Divide(Vector2 v1, Vector2 v2); /// Normalize provided vector [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Normalize(Vector2 v); /// Transforms a Vector2 by a given Matrix [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Transform(Vector2 v, Matrix4x4 mat); /// Calculate linear interpolation between two vectors [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Lerp(Vector2 v1, Vector2 v2, float amount); /// Calculate reflected vector to normal [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Reflect(Vector2 v, Vector2 normal); /// Rotate vector by angle [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Rotate(Vector2 v, float angle); /// Move Vector towards target [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2MoveTowards(Vector2 v, Vector2 target, float maxDistance); /// Invert the given vector [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Invert(Vector2 v); /// /// Clamp the components of the vector between min and max values specified by the given vectors /// [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Clamp(Vector2 v, Vector2 min, Vector2 max); /// Clamp the magnitude of the vector between two min and max values [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2ClampValue(Vector2 v, float min, float max); /// Check whether two given vectors are almost equal [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern int Vector2Equals(Vector2 p, Vector2 q); /// Compute the direction of a refracted ray /// normalized direction of the incoming ray /// normalized normal vector of the interface of two optical media /// /// ratio of the refractive index of the medium from where the ray comes /// to the refractive index of the medium on the other side of the surface /// [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector2Refract(Vector2 v, Vector2 n, float r); /// Vector with components value 0.0f [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Zero(); /// Vector with components value 1.0f [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3One(); /// Add two vectors [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Add(Vector3 v1, Vector3 v2); /// Add vector and float value [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3AddValue(Vector3 v, float add); /// Subtract two vectors [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Subtract(Vector3 v1, Vector3 v2); /// Subtract vector and float value [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3SubtractValue(Vector3 v, float sub); /// Multiply vector by scalar [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Scale(Vector3 v, float scalar); /// Multiply vector by vector [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Multiply(Vector3 v1, Vector3 v2); /// Calculate two vectors cross product [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3CrossProduct(Vector3 v1, Vector3 v2); /// Calculate one vector perpendicular vector [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Perpendicular(Vector3 v); /// Calculate vector length [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Vector3Length(Vector3 v); /// Calculate vector square length [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Vector3LengthSqr(Vector3 v); /// Calculate two vectors dot product [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Vector3DotProduct(Vector3 v1, Vector3 v2); /// Calculate distance between two vectors [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Vector3Distance(Vector3 v1, Vector3 v2); /// Calculate square distance between two vectors [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float Vector3DistanceSqr(Vector3 v1, Vector3 v2); /// Calculate angle between two vectors in XY and XZ [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector2 Vector3Angle(Vector3 v1, Vector3 v2); /// Negate provided vector (invert direction) [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Negate(Vector3 v); /// Divide vector by vector [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Divide(Vector3 v1, Vector3 v2); /// Normalize provided vector [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Normalize(Vector3 v); /// Calculate the projection of the vector v1 on to v2 [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Project(Vector3 v1, Vector3 v2); /// Calculate the rejection of the vector v1 on to v2 [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Reject(Vector3 v1, Vector3 v2); /// /// Orthonormalize provided vectors
/// Makes vectors normalized and orthogonal to each other
/// Gram-Schmidt function implementation ///
[DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern void Vector3OrthoNormalize(Vector3* v1, Vector3* v2); /// Transforms a Vector3 by a given Matrix [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Transform(Vector3 v, Matrix4x4 mat); /// Transform a vector by quaternion rotation [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3RotateByQuaternion(Vector3 v, Quaternion q); /// Rotates a vector around an axis [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3RotateByAxisAngle(Vector3 v, Vector3 axis, float angle); /// Move Vector towards target [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3MoveTowards(Vector3 v, Vector3 target, float maxDistance); /// Calculate linear interpolation between two vectors [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Lerp(Vector3 v1, Vector3 v2, float amount); /// /// Calculate cubic hermite interpolation between two vectors and their tangents /// as described in the GLTF 2.0 specification: https://registry.khronos.org/glTF/specs/2.0/glTF-2.0.html#interpolation-cubic /// [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3CubicHermite(Vector3 v1, Vector3 tangent1, Vector3 v2, Vector3 tangent2, float amount); /// Calculate reflected vector to normal [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Reflect(Vector3 v, Vector3 normal); /// Get min value for each pair of components [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Min(Vector3 v1, Vector3 v2); /// Get max value for each pair of components [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Max(Vector3 v1, Vector3 v2); /// /// Compute barycenter coordinates (u, v, w) for point p with respect to triangle (a, b, c)
/// NOTE: Assumes P is on the plane of the triangle ///
[DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Barycenter(Vector3 p, Vector3 a, Vector3 b, Vector3 c); /// /// Projects a Vector3 from screen space into object space
/// NOTE: We are avoiding calling other raymath functions despite available ///
[DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Unproject(Vector3 source, Matrix4x4 projection, Matrix4x4 view); /// Get Vector3 as float array [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Float3 Vector3ToFloatV(Vector3 v); /// Invert the given vector [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Invert(Vector3 v); /// /// Clamp the components of the vector between /// min and max values specified by the given vectors /// [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Clamp(Vector3 v, Vector3 min, Vector3 max); /// Clamp the magnitude of the vector between two values [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3ClampValue(Vector3 v, float min, float max); /// Check whether two given vectors are almost equal [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern int Vector3Equals(Vector3 p, Vector3 q); /// /// Compute the direction of a refracted ray where v specifies the /// normalized direction of the incoming ray, n specifies the /// normalized normal vector of the interface of two optical media, /// and r specifies the ratio of the refractive index of the medium /// from where the ray comes to the refractive index of the medium /// on the other side of the surface /// [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 Vector3Refract(Vector3 v, Vector3 n, float r); /// Compute matrix determinant [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float MatrixDeterminant(Matrix4x4 mat); /// Get the trace of the matrix (sum of the values along the diagonal) [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float MatrixTrace(Matrix4x4 mat); /// Transposes provided matrix [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixTranspose(Matrix4x4 mat); /// Invert provided matrix [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixInvert(Matrix4x4 mat); /// Get identity matrix [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixIdentity(); /// Add two matrices [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixAdd(Matrix4x4 left, Matrix4x4 right); /// Subtract two matrices (left - right) [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixSubtract(Matrix4x4 left, Matrix4x4 right); /// /// Get two matrix multiplication
/// NOTE: When multiplying matrices... the order matters! ///
[DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixMultiply(Matrix4x4 left, Matrix4x4 right); /// Get translation matrix [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixTranslate(float x, float y, float z); /// /// Create rotation matrix from axis and angle
/// NOTE: Angle should be provided in radians ///
[DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixRotate(Vector3 axis, float angle); /// Get x-rotation matrix (angle in radians) [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixRotateX(float angle); /// Get y-rotation matrix (angle in radians) [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixRotateY(float angle); /// Get z-rotation matrix (angle in radians) [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixRotateZ(float angle); /// Get xyz-rotation matrix (angles in radians) [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixRotateXYZ(Vector3 ang); /// Get zyx-rotation matrix (angles in radians) [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixRotateZYX(Vector3 ang); /// Get scaling matrix [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixScale(float x, float y, float z); /// Get perspective projection matrix [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixFrustum( double left, double right, double bottom, double top, double nearPlane, double farPlane ); /// /// Get perspective projection matrix
/// NOTE: Angle should be provided in radians ///
[DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixPerspective(double fovy, double aspect, double near, double far); /// Get orthographic projection matrix [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixOrtho( double left, double right, double bottom, double top, double near, double far ); /// Get camera look-at matrix (view matrix) [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 MatrixLookAt(Vector3 eye, Vector3 target, Vector3 up); /// Get float array of matrix data [DllImport(Raylib.NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Float16 MatrixToFloatV(Matrix4x4 m); /// Add 2 quaternions [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionAdd(Quaternion q1, Quaternion q2); /// Add quaternion and float value [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionAddValue(Quaternion q, float add); /// Subtract 2 quaternions [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionSubtract(Quaternion q1, Quaternion q2); /// Subtract quaternion and float value [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionSubtractValue(Quaternion q, float add); /// Get identity quaternion [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionIdentity(); /// Computes the length of a quaternion [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern float QuaternionLength(Quaternion q); /// Normalize provided quaternion [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionNormalize(Quaternion q); /// Invert provided quaternion [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionInvert(Quaternion q); /// Calculate two quaternion multiplication [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionMultiply(Quaternion q1, Quaternion q2); /// Scale quaternion by float value [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionScale(Quaternion q, float mul); /// Divide two quaternions [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionDivide(Quaternion q1, Quaternion q2); /// Calculate linear interpolation between two quaternions [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionLerp(Quaternion q1, Quaternion q2, float amount); /// Calculate slerp-optimized interpolation between two quaternions [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionNlerp(Quaternion q1, Quaternion q2, float amount); /// Calculates spherical linear interpolation between two quaternions [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionSlerp(Quaternion q1, Quaternion q2, float amount); /// /// Calculate quaternion cubic spline interpolation using Cubic Hermite Spline algorithm /// as described in the GLTF 2.0 specification: https://registry.khronos.org/glTF/specs/2.0/glTF-2.0.html#interpolation-cubic /// [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionCubicHermiteSpline(Quaternion q1, Quaternion outTangent1, Quaternion q2, Quaternion inTangent2, float t); /// Calculate quaternion based on the rotation from one vector to another [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionFromVector3ToVector3(Vector3 from, Vector3 to); /// Get a quaternion for a given rotation matrix [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionFromMatrix(Matrix4x4 mat); /// Get a matrix for a given quaternion [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Matrix4x4 QuaternionToMatrix(Quaternion q); /// /// Get rotation quaternion for an angle and axis
/// NOTE: angle must be provided in radians ///
[DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionFromAxisAngle(Vector3 axis, float angle); /// Get the rotation angle and axis for a given quaternion [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern void QuaternionToAxisAngle(Quaternion q, Vector3* outAxis, float* outAngle); /// /// Get the quaternion equivalent to Euler angles
/// NOTE: Rotation order is ZYX ///
[DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionFromEuler(float pitch, float yaw, float roll); /// /// Get the Euler angles equivalent to quaternion (roll, pitch, yaw)
/// NOTE: Angles are returned in a Vector3 struct in radians ///
[DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Vector3 QuaternionToEuler(Quaternion q); /// Transform a quaternion given a transformation matrix [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern Quaternion QuaternionTransform(Quaternion q, Matrix4x4 mat); /// Check whether two given quaternions are almost equal [DllImport(NativeLibName, CallingConvention = CallingConvention.Cdecl)] public static extern int QuaternionEquals(Quaternion p, Quaternion q); }