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