| Adrian Taylor | c871343 | 2020-10-21 18:20:55 -0700 | [diff] [blame^] | 1 | use crate::syntax::{ |
| 2 | Api, CppName, ExternFn, Impl, Namespace, Pair, Receiver, Ref, ResolvableName, Signature, Slice, |
| 3 | Symbol, Ty1, Type, Types, |
| 4 | }; |
| 5 | use proc_macro2::{Ident, Span}; |
| David Tolnay | 6586c69 | 2020-10-03 23:48:19 -0700 | [diff] [blame] | 6 | use std::borrow::Borrow; |
| David Tolnay | 7db7369 | 2019-10-20 14:51:12 -0400 | [diff] [blame] | 7 | use std::hash::{Hash, Hasher}; |
| David Tolnay | c21b20a | 2020-03-15 23:25:32 -0700 | [diff] [blame] | 8 | use std::mem; |
| David Tolnay | 9bfbea3 | 2020-04-22 18:12:43 -0700 | [diff] [blame] | 9 | use std::ops::{Deref, DerefMut}; |
| Adrian Taylor | c871343 | 2020-10-21 18:20:55 -0700 | [diff] [blame^] | 10 | use syn::Token; |
| David Tolnay | 1644873 | 2020-03-18 12:39:36 -0700 | [diff] [blame] | 11 | |
| 12 | impl Deref for ExternFn { |
| 13 | type Target = Signature; |
| 14 | |
| 15 | fn deref(&self) -> &Self::Target { |
| 16 | &self.sig |
| 17 | } |
| 18 | } |
| David Tolnay | c21b20a | 2020-03-15 23:25:32 -0700 | [diff] [blame] | 19 | |
| David Tolnay | 9bfbea3 | 2020-04-22 18:12:43 -0700 | [diff] [blame] | 20 | impl DerefMut for ExternFn { |
| 21 | fn deref_mut(&mut self) -> &mut Self::Target { |
| 22 | &mut self.sig |
| 23 | } |
| 24 | } |
| 25 | |
| David Tolnay | c21b20a | 2020-03-15 23:25:32 -0700 | [diff] [blame] | 26 | impl Hash for Type { |
| 27 | fn hash<H: Hasher>(&self, state: &mut H) { |
| 28 | mem::discriminant(self).hash(state); |
| 29 | match self { |
| 30 | Type::Ident(t) => t.hash(state), |
| 31 | Type::RustBox(t) => t.hash(state), |
| 32 | Type::UniquePtr(t) => t.hash(state), |
| 33 | Type::Ref(t) => t.hash(state), |
| 34 | Type::Str(t) => t.hash(state), |
| Myron Ahn | eba35cf | 2020-02-05 19:41:51 +0700 | [diff] [blame] | 35 | Type::RustVec(t) => t.hash(state), |
| David Tolnay | 4377a9e | 2020-04-24 15:20:26 -0700 | [diff] [blame] | 36 | Type::CxxVector(t) => t.hash(state), |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 37 | Type::Fn(t) => t.hash(state), |
| Adrian Taylor | f5dd552 | 2020-04-13 16:50:14 -0700 | [diff] [blame] | 38 | Type::Slice(t) => t.hash(state), |
| 39 | Type::SliceRefU8(t) => t.hash(state), |
| David Tolnay | c21b20a | 2020-03-15 23:25:32 -0700 | [diff] [blame] | 40 | Type::Void(_) => {} |
| 41 | } |
| 42 | } |
| 43 | } |
| 44 | |
| 45 | impl Eq for Type {} |
| 46 | |
| 47 | impl PartialEq for Type { |
| 48 | fn eq(&self, other: &Type) -> bool { |
| 49 | match (self, other) { |
| 50 | (Type::Ident(lhs), Type::Ident(rhs)) => lhs == rhs, |
| 51 | (Type::RustBox(lhs), Type::RustBox(rhs)) => lhs == rhs, |
| 52 | (Type::UniquePtr(lhs), Type::UniquePtr(rhs)) => lhs == rhs, |
| 53 | (Type::Ref(lhs), Type::Ref(rhs)) => lhs == rhs, |
| 54 | (Type::Str(lhs), Type::Str(rhs)) => lhs == rhs, |
| Myron Ahn | eba35cf | 2020-02-05 19:41:51 +0700 | [diff] [blame] | 55 | (Type::RustVec(lhs), Type::RustVec(rhs)) => lhs == rhs, |
| David Tolnay | 4377a9e | 2020-04-24 15:20:26 -0700 | [diff] [blame] | 56 | (Type::CxxVector(lhs), Type::CxxVector(rhs)) => lhs == rhs, |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 57 | (Type::Fn(lhs), Type::Fn(rhs)) => lhs == rhs, |
| Adrian Taylor | f5dd552 | 2020-04-13 16:50:14 -0700 | [diff] [blame] | 58 | (Type::Slice(lhs), Type::Slice(rhs)) => lhs == rhs, |
| 59 | (Type::SliceRefU8(lhs), Type::SliceRefU8(rhs)) => lhs == rhs, |
| David Tolnay | c21b20a | 2020-03-15 23:25:32 -0700 | [diff] [blame] | 60 | (Type::Void(_), Type::Void(_)) => true, |
| 61 | (_, _) => false, |
| 62 | } |
| 63 | } |
| 64 | } |
| David Tolnay | 7db7369 | 2019-10-20 14:51:12 -0400 | [diff] [blame] | 65 | |
| 66 | impl Eq for Ty1 {} |
| 67 | |
| 68 | impl PartialEq for Ty1 { |
| 69 | fn eq(&self, other: &Ty1) -> bool { |
| David Tolnay | 35c82eb | 2020-03-18 16:48:28 -0700 | [diff] [blame] | 70 | let Ty1 { |
| 71 | name, |
| 72 | langle: _, |
| 73 | inner, |
| 74 | rangle: _, |
| 75 | } = self; |
| 76 | let Ty1 { |
| 77 | name: name2, |
| 78 | langle: _, |
| 79 | inner: inner2, |
| 80 | rangle: _, |
| 81 | } = other; |
| 82 | name == name2 && inner == inner2 |
| David Tolnay | 7db7369 | 2019-10-20 14:51:12 -0400 | [diff] [blame] | 83 | } |
| 84 | } |
| 85 | |
| 86 | impl Hash for Ty1 { |
| 87 | fn hash<H: Hasher>(&self, state: &mut H) { |
| David Tolnay | 35c82eb | 2020-03-18 16:48:28 -0700 | [diff] [blame] | 88 | let Ty1 { |
| 89 | name, |
| 90 | langle: _, |
| 91 | inner, |
| 92 | rangle: _, |
| 93 | } = self; |
| 94 | name.hash(state); |
| 95 | inner.hash(state); |
| David Tolnay | 7db7369 | 2019-10-20 14:51:12 -0400 | [diff] [blame] | 96 | } |
| 97 | } |
| 98 | |
| 99 | impl Eq for Ref {} |
| 100 | |
| 101 | impl PartialEq for Ref { |
| 102 | fn eq(&self, other: &Ref) -> bool { |
| David Tolnay | 35c82eb | 2020-03-18 16:48:28 -0700 | [diff] [blame] | 103 | let Ref { |
| 104 | ampersand: _, |
| David Tolnay | 0bd50fa | 2020-04-22 15:31:33 -0700 | [diff] [blame] | 105 | lifetime, |
| David Tolnay | 35c82eb | 2020-03-18 16:48:28 -0700 | [diff] [blame] | 106 | mutability, |
| 107 | inner, |
| 108 | } = self; |
| 109 | let Ref { |
| 110 | ampersand: _, |
| David Tolnay | 0bd50fa | 2020-04-22 15:31:33 -0700 | [diff] [blame] | 111 | lifetime: lifetime2, |
| David Tolnay | 35c82eb | 2020-03-18 16:48:28 -0700 | [diff] [blame] | 112 | mutability: mutability2, |
| 113 | inner: inner2, |
| 114 | } = other; |
| David Tolnay | 0bd50fa | 2020-04-22 15:31:33 -0700 | [diff] [blame] | 115 | lifetime == lifetime2 && mutability.is_some() == mutability2.is_some() && inner == inner2 |
| David Tolnay | 7db7369 | 2019-10-20 14:51:12 -0400 | [diff] [blame] | 116 | } |
| 117 | } |
| 118 | |
| 119 | impl Hash for Ref { |
| 120 | fn hash<H: Hasher>(&self, state: &mut H) { |
| David Tolnay | 35c82eb | 2020-03-18 16:48:28 -0700 | [diff] [blame] | 121 | let Ref { |
| 122 | ampersand: _, |
| David Tolnay | 0bd50fa | 2020-04-22 15:31:33 -0700 | [diff] [blame] | 123 | lifetime, |
| David Tolnay | 35c82eb | 2020-03-18 16:48:28 -0700 | [diff] [blame] | 124 | mutability, |
| 125 | inner, |
| 126 | } = self; |
| David Tolnay | 0bd50fa | 2020-04-22 15:31:33 -0700 | [diff] [blame] | 127 | lifetime.hash(state); |
| David Tolnay | 35c82eb | 2020-03-18 16:48:28 -0700 | [diff] [blame] | 128 | mutability.is_some().hash(state); |
| 129 | inner.hash(state); |
| David Tolnay | 7db7369 | 2019-10-20 14:51:12 -0400 | [diff] [blame] | 130 | } |
| 131 | } |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 132 | |
| Adrian Taylor | f5dd552 | 2020-04-13 16:50:14 -0700 | [diff] [blame] | 133 | impl Eq for Slice {} |
| 134 | |
| 135 | impl PartialEq for Slice { |
| 136 | fn eq(&self, other: &Slice) -> bool { |
| David Tolnay | eb952ba | 2020-04-14 15:02:24 -0700 | [diff] [blame] | 137 | let Slice { bracket: _, inner } = self; |
| Adrian Taylor | f5dd552 | 2020-04-13 16:50:14 -0700 | [diff] [blame] | 138 | let Slice { |
| 139 | bracket: _, |
| 140 | inner: inner2, |
| 141 | } = other; |
| 142 | inner == inner2 |
| 143 | } |
| 144 | } |
| 145 | |
| 146 | impl Hash for Slice { |
| 147 | fn hash<H: Hasher>(&self, state: &mut H) { |
| David Tolnay | eb952ba | 2020-04-14 15:02:24 -0700 | [diff] [blame] | 148 | let Slice { bracket: _, inner } = self; |
| Adrian Taylor | f5dd552 | 2020-04-13 16:50:14 -0700 | [diff] [blame] | 149 | inner.hash(state); |
| 150 | } |
| 151 | } |
| 152 | |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 153 | impl Eq for Signature {} |
| 154 | |
| 155 | impl PartialEq for Signature { |
| 156 | fn eq(&self, other: &Signature) -> bool { |
| 157 | let Signature { |
| David Tolnay | bdb576c | 2020-09-06 23:45:55 -0700 | [diff] [blame] | 158 | unsafety, |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 159 | fn_token: _, |
| 160 | receiver, |
| 161 | args, |
| 162 | ret, |
| 163 | throws, |
| David Tolnay | e3a4815 | 2020-04-08 19:38:05 -0700 | [diff] [blame] | 164 | paren_token: _, |
| 165 | throws_tokens: _, |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 166 | } = self; |
| 167 | let Signature { |
| David Tolnay | bdb576c | 2020-09-06 23:45:55 -0700 | [diff] [blame] | 168 | unsafety: unsafety2, |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 169 | fn_token: _, |
| 170 | receiver: receiver2, |
| 171 | args: args2, |
| 172 | ret: ret2, |
| 173 | throws: throws2, |
| David Tolnay | e3a4815 | 2020-04-08 19:38:05 -0700 | [diff] [blame] | 174 | paren_token: _, |
| 175 | throws_tokens: _, |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 176 | } = other; |
| David Tolnay | bdb576c | 2020-09-06 23:45:55 -0700 | [diff] [blame] | 177 | unsafety.is_some() == unsafety2.is_some() |
| 178 | && receiver == receiver2 |
| David Tolnay | e3a4815 | 2020-04-08 19:38:05 -0700 | [diff] [blame] | 179 | && ret == ret2 |
| 180 | && throws == throws2 |
| 181 | && args.len() == args2.len() |
| 182 | && args.iter().zip(args2).all(|(arg, arg2)| arg == arg2) |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 183 | } |
| 184 | } |
| 185 | |
| 186 | impl Hash for Signature { |
| 187 | fn hash<H: Hasher>(&self, state: &mut H) { |
| 188 | let Signature { |
| David Tolnay | bdb576c | 2020-09-06 23:45:55 -0700 | [diff] [blame] | 189 | unsafety, |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 190 | fn_token: _, |
| 191 | receiver, |
| 192 | args, |
| 193 | ret, |
| 194 | throws, |
| David Tolnay | e3a4815 | 2020-04-08 19:38:05 -0700 | [diff] [blame] | 195 | paren_token: _, |
| 196 | throws_tokens: _, |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 197 | } = self; |
| David Tolnay | bdb576c | 2020-09-06 23:45:55 -0700 | [diff] [blame] | 198 | unsafety.is_some().hash(state); |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 199 | receiver.hash(state); |
| David Tolnay | e3a4815 | 2020-04-08 19:38:05 -0700 | [diff] [blame] | 200 | for arg in args { |
| 201 | arg.hash(state); |
| 202 | } |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 203 | ret.hash(state); |
| 204 | throws.hash(state); |
| 205 | } |
| 206 | } |
| 207 | |
| 208 | impl Eq for Receiver {} |
| 209 | |
| 210 | impl PartialEq for Receiver { |
| 211 | fn eq(&self, other: &Receiver) -> bool { |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 212 | let Receiver { |
| David Tolnay | fb6e386 | 2020-04-20 01:33:23 -0700 | [diff] [blame] | 213 | ampersand: _, |
| David Tolnay | 0bd50fa | 2020-04-22 15:31:33 -0700 | [diff] [blame] | 214 | lifetime, |
| David Tolnay | fb6e386 | 2020-04-20 01:33:23 -0700 | [diff] [blame] | 215 | mutability, |
| David Tolnay | 05e11cc | 2020-04-20 02:13:56 -0700 | [diff] [blame] | 216 | var: _, |
| 217 | ty, |
| David Tolnay | 62d360c | 2020-04-22 16:26:21 -0700 | [diff] [blame] | 218 | shorthand: _, |
| David Tolnay | fb6e386 | 2020-04-20 01:33:23 -0700 | [diff] [blame] | 219 | } = self; |
| 220 | let Receiver { |
| 221 | ampersand: _, |
| David Tolnay | 0bd50fa | 2020-04-22 15:31:33 -0700 | [diff] [blame] | 222 | lifetime: lifetime2, |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 223 | mutability: mutability2, |
| David Tolnay | 05e11cc | 2020-04-20 02:13:56 -0700 | [diff] [blame] | 224 | var: _, |
| 225 | ty: ty2, |
| David Tolnay | 62d360c | 2020-04-22 16:26:21 -0700 | [diff] [blame] | 226 | shorthand: _, |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 227 | } = other; |
| David Tolnay | 0bd50fa | 2020-04-22 15:31:33 -0700 | [diff] [blame] | 228 | lifetime == lifetime2 && mutability.is_some() == mutability2.is_some() && ty == ty2 |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 229 | } |
| 230 | } |
| 231 | |
| 232 | impl Hash for Receiver { |
| 233 | fn hash<H: Hasher>(&self, state: &mut H) { |
| David Tolnay | fb6e386 | 2020-04-20 01:33:23 -0700 | [diff] [blame] | 234 | let Receiver { |
| 235 | ampersand: _, |
| David Tolnay | 0bd50fa | 2020-04-22 15:31:33 -0700 | [diff] [blame] | 236 | lifetime, |
| David Tolnay | fb6e386 | 2020-04-20 01:33:23 -0700 | [diff] [blame] | 237 | mutability, |
| David Tolnay | 05e11cc | 2020-04-20 02:13:56 -0700 | [diff] [blame] | 238 | var: _, |
| 239 | ty, |
| David Tolnay | 62d360c | 2020-04-22 16:26:21 -0700 | [diff] [blame] | 240 | shorthand: _, |
| David Tolnay | fb6e386 | 2020-04-20 01:33:23 -0700 | [diff] [blame] | 241 | } = self; |
| David Tolnay | 0bd50fa | 2020-04-22 15:31:33 -0700 | [diff] [blame] | 242 | lifetime.hash(state); |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 243 | mutability.is_some().hash(state); |
| David Tolnay | 05e11cc | 2020-04-20 02:13:56 -0700 | [diff] [blame] | 244 | ty.hash(state); |
| David Tolnay | 417305a | 2020-03-18 13:54:00 -0700 | [diff] [blame] | 245 | } |
| 246 | } |
| David Tolnay | 6586c69 | 2020-10-03 23:48:19 -0700 | [diff] [blame] | 247 | |
| 248 | impl Hash for Impl { |
| 249 | fn hash<H: Hasher>(&self, state: &mut H) { |
| 250 | let Impl { |
| 251 | impl_token: _, |
| 252 | ty, |
| 253 | brace_token: _, |
| 254 | } = self; |
| 255 | ty.hash(state); |
| 256 | } |
| 257 | } |
| 258 | |
| 259 | impl Eq for Impl {} |
| 260 | |
| 261 | impl PartialEq for Impl { |
| 262 | fn eq(&self, other: &Impl) -> bool { |
| 263 | let Impl { |
| 264 | impl_token: _, |
| 265 | ty, |
| 266 | brace_token: _, |
| 267 | } = self; |
| 268 | let Impl { |
| 269 | impl_token: _, |
| 270 | ty: ty2, |
| 271 | brace_token: _, |
| 272 | } = other; |
| 273 | ty == ty2 |
| 274 | } |
| 275 | } |
| 276 | |
| 277 | impl Borrow<Type> for &Impl { |
| 278 | fn borrow(&self) -> &Type { |
| 279 | &self.ty |
| 280 | } |
| 281 | } |
| Adrian Taylor | c871343 | 2020-10-21 18:20:55 -0700 | [diff] [blame^] | 282 | |
| 283 | impl Pair { |
| 284 | /// Use this constructor when the item can't have a different |
| 285 | /// name in Rust and C++. For cases where #[rust_name] and similar |
| 286 | /// attributes can be used, construct the object by hand. |
| 287 | pub fn new(ns: Namespace, ident: Ident) -> Self { |
| 288 | Self { |
| 289 | rust: ident.clone(), |
| 290 | cxx: CppName::new(ns, ident), |
| 291 | } |
| 292 | } |
| 293 | } |
| 294 | |
| 295 | impl ResolvableName { |
| 296 | pub fn new(ident: Ident) -> Self { |
| 297 | Self { rust: ident } |
| 298 | } |
| 299 | |
| 300 | pub fn from_pair(pair: Pair) -> Self { |
| 301 | Self { rust: pair.rust } |
| 302 | } |
| 303 | |
| 304 | pub fn make_self(span: Span) -> Self { |
| 305 | Self { |
| 306 | rust: Token.into(), |
| 307 | } |
| 308 | } |
| 309 | |
| 310 | pub fn is_self(&self) -> bool { |
| 311 | self.rust == "Self" |
| 312 | } |
| 313 | |
| 314 | pub fn span(&self) -> Span { |
| 315 | self.rust.span() |
| 316 | } |
| 317 | |
| 318 | pub fn to_symbol(&self, types: &Types) -> Symbol { |
| 319 | types.resolve(self).to_symbol() |
| 320 | } |
| 321 | } |
| 322 | |
| 323 | impl Api { |
| 324 | pub fn get_namespace(&self) -> Option<&Namespace> { |
| 325 | match self { |
| 326 | Api::CxxFunction(cfn) => Some(&cfn.ident.cxx.ns), |
| 327 | Api::CxxType(cty) => Some(&cty.ident.cxx.ns), |
| 328 | Api::Enum(enm) => Some(&enm.ident.cxx.ns), |
| 329 | Api::Struct(strct) => Some(&strct.ident.cxx.ns), |
| 330 | Api::RustType(rty) => Some(&rty.ident.cxx.ns), |
| 331 | Api::RustFunction(rfn) => Some(&rfn.ident.cxx.ns), |
| 332 | Api::Impl(_) | Api::Include(_) | Api::TypeAlias(_) => None, |
| 333 | } |
| 334 | } |
| 335 | } |
| 336 | |
| 337 | impl CppName { |
| 338 | pub fn new(ns: Namespace, ident: Ident) -> Self { |
| 339 | Self { ns, ident } |
| 340 | } |
| 341 | |
| 342 | fn iter_all_segments( |
| 343 | &self, |
| 344 | ) -> std::iter::Chain<std::slice::Iter<Ident>, std::iter::Once<&Ident>> { |
| 345 | self.ns.iter().chain(std::iter::once(&self.ident)) |
| 346 | } |
| 347 | |
| 348 | fn join(&self, sep: &str) -> String { |
| 349 | self.iter_all_segments() |
| 350 | .map(|s| s.to_string()) |
| 351 | .collect::<Vec<_>>() |
| 352 | .join(sep) |
| 353 | } |
| 354 | |
| 355 | pub fn to_symbol(&self) -> Symbol { |
| 356 | Symbol::from_idents(self.iter_all_segments()) |
| 357 | } |
| 358 | |
| 359 | pub fn to_fully_qualified(&self) -> String { |
| 360 | format!("::{}", self.join("::")) |
| 361 | } |
| 362 | } |