Coverage for pySDC/implementations/datatype_classes/particles.py: 79%
112 statements
« prev ^ index » next coverage.py v7.6.1, created at 2024-09-09 14:59 +0000
« prev ^ index » next coverage.py v7.6.1, created at 2024-09-09 14:59 +0000
1import numpy as np
3from pySDC.implementations.datatype_classes.mesh import mesh
4from pySDC.core.errors import DataError
6try:
7 from mpi4py import MPI
8except ImportError:
9 MPI = None
12class particles(object):
13 """
14 Particle data type for particles in 3 dimensions
16 This data type can be used for particles in 3 dimensions with 3 position and 3 velocity values per particle
18 Attributes:
19 pos: contains the positions of all particles
20 vel: contains the velocities of all particles
21 """
23 class position(mesh):
24 pass
26 class velocity(mesh):
27 pass
29 def __init__(self, init=None, val=None):
30 """
31 Initialization routine
33 Args:
34 init: can either be a number or another particle object
35 val: initial tuple of values for position and velocity (default: (None,None))
36 Raises:
37 DataError: if init is none of the types above
38 """
40 # if init is another particles object, do a copy (init by copy)
41 if isinstance(init, type(self)):
42 self.pos = particles.position(init.pos)
43 self.vel = particles.velocity(init.vel)
44 self.q = init.q.copy()
45 self.m = init.m.copy()
46 # if init is a number, create particles object and pick the corresponding initial values
47 elif (
48 isinstance(init, tuple)
49 and (init[1] is None or isinstance(init[1], MPI.Intracomm))
50 and isinstance(init[2], np.dtype)
51 ):
52 if isinstance(val, int) or isinstance(val, float) or val is None:
53 self.pos = particles.position(init, val=val)
54 self.vel = particles.velocity(init, val=val)
55 if isinstance(init[0], tuple):
56 self.q = np.zeros(init[0][-1])
57 self.m = np.zeros(init[0][-1])
58 elif isinstance(init[0], int):
59 self.q = np.zeros(init[0])
60 self.m = np.zeros(init[0])
61 self.q[:] = 1.0
62 self.m[:] = 1.0
63 elif isinstance(val, tuple) and len(val) == 4:
64 self.pos = particles.position(init, val=val[0])
65 self.vel = particles.velocity(init, val=val[1])
66 if isinstance(init[0], tuple):
67 self.q = np.zeros(init[0][-1])
68 self.m = np.zeros(init[0][-1])
69 elif isinstance(init[0], int):
70 self.q = np.zeros(init[0])
71 self.m = np.zeros(init[0])
72 self.q[:] = val[2]
73 self.m[:] = val[3]
74 else:
75 raise DataError('type of val is wrong, got %s', val)
76 # something is wrong, if none of the ones above hit
77 else:
78 raise DataError('something went wrong during %s initialization' % type(self))
80 def __add__(self, other):
81 """
82 Overloading the addition operator for particles types
84 Args:
85 other (particles): particles object to be added
86 Raises:
87 DataError: if other is not a particles object
88 Returns:
89 particles: sum of caller and other values (self+other)
90 """
92 if isinstance(other, type(self)):
93 # always create new particles, since otherwise c = a + b changes a as well!
94 p = particles(self)
95 p.pos[:] = self.pos + other.pos
96 p.vel[:] = self.vel + other.vel
97 p.m = self.m
98 p.q = self.q
99 return p
100 else:
101 raise DataError("Type error: cannot add %s to %s" % (type(other), type(self)))
103 def __sub__(self, other):
104 """
105 Overloading the subtraction operator for particles types
107 Args:
108 other (particles): particles object to be subtracted
109 Raises:
110 DataError: if other is not a particles object
111 Returns:
112 particles: differences between caller and other values (self-other)
113 """
115 if isinstance(other, type(self)):
116 # always create new particles, since otherwise c = a - b changes a as well!
117 p = particles(self)
118 p.pos[:] = self.pos - other.pos
119 p.vel[:] = self.vel - other.vel
120 p.m = self.m
121 p.q = self.q
122 return p
123 else:
124 raise DataError("Type error: cannot subtract %s from %s" % (type(other), type(self)))
126 def __rmul__(self, other):
127 """
128 Overloading the right multiply by factor operator for particles types
130 Args:
131 other (float): factor
132 Raises:
133 DataError: if other is not a particles object
134 Returns:
135 particles: scaled particle's velocity and position as new particle
136 """
138 if isinstance(other, float):
139 # always create new particles
140 p = particles(self)
141 p.pos[:] = other * self.pos
142 p.vel[:] = other * self.vel
143 p.m = self.m
144 p.q = self.q
145 return p
146 else:
147 raise DataError("Type error: cannot multiply %s to %s" % (type(other), type(self)))
149 def __abs__(self):
150 """
151 Overloading the abs operator for particles types
153 Returns:
154 float: absolute maximum of abs(pos) and abs(vel) for all particles
155 """
156 abspos = abs(self.pos)
157 absvel = abs(self.vel)
158 return np.amax((abspos, absvel))
160 def send(self, dest=None, tag=None, comm=None):
161 """
162 Routine for sending data forward in time (blocking)
164 Args:
165 dest (int): target rank
166 tag (int): communication tag
167 comm: communicator
169 Returns:
170 None
171 """
173 comm.send(self, dest=dest, tag=tag)
174 return None
176 def isend(self, dest=None, tag=None, comm=None):
177 """
178 Routine for sending data forward in time (non-blocking)
180 Args:
181 dest (int): target rank
182 tag (int): communication tag
183 comm: communicator
185 Returns:
186 request handle
187 """
188 return comm.isend(self, dest=dest, tag=tag)
190 def recv(self, source=None, tag=None, comm=None):
191 """
192 Routine for receiving in time
194 Args:
195 source (int): source rank
196 tag (int): communication tag
197 comm: communicator
199 Returns:
200 None
201 """
202 part = comm.recv(source=source, tag=tag)
203 self.pos[:] = part.pos.copy()
204 self.vel[:] = part.vel.copy()
205 self.m = part.m.copy()
206 self.q = part.q.copy()
207 return None
210class acceleration(mesh):
211 pass
214class fields(object):
215 """
216 Field data type for 3 dimensions
218 This data type can be used for electric and magnetic fields in 3 dimensions
220 Attributes:
221 elec: contains the electric field
222 magn: contains the magnetic field
223 """
225 class electric(mesh):
226 pass
228 class magnetic(mesh):
229 pass
231 def __init__(self, init=None, val=None):
232 """
233 Initialization routine
235 Args:
236 init: can either be a number or another fields object
237 val: initial tuple of values for electric and magnetic (default: (None,None))
238 Raises:
239 DataError: if init is none of the types above
240 """
242 # if init is another fields object, do a copy (init by copy)
243 if isinstance(init, type(self)):
244 self.elec = fields.electric(init.elec)
245 self.magn = fields.magnetic(init.magn)
246 # if init is a number, create fields object and pick the corresponding initial values
247 elif (
248 isinstance(init, tuple)
249 and (init[1] is None or isinstance(init[1], MPI.Intracomm))
250 and isinstance(init[2], np.dtype)
251 ):
252 if isinstance(val, int) or isinstance(val, float) or val is None:
253 self.elec = fields.electric(init, val=val)
254 self.magn = fields.magnetic(init, val=val)
255 elif isinstance(val, tuple) and len(val) == 2:
256 self.elec = fields.electric(init, val=val[0])
257 self.magn = fields.magnetic(init, val=val[1])
258 else:
259 raise DataError('wrong type of val, got %s' % val)
260 # something is wrong, if none of the ones above hit
261 else:
262 raise DataError('something went wrong during %s initialization' % type(self))
264 def __add__(self, other):
265 """
266 Overloading the addition operator for fields types
268 Args:
269 other (fields): fields object to be added
270 Raises:
271 DataError: if other is not a fields object
272 Returns:
273 fields: sum of caller and other values (self+other)
274 """
276 if isinstance(other, type(self)):
277 # always create new fields, since otherwise c = a - b changes a as well!
278 p = fields(self)
279 p.elec[:] = self.elec + other.elec
280 p.magn[:] = self.magn + other.magn
281 return p
282 else:
283 raise DataError("Type error: cannot add %s to %s" % (type(other), type(self)))
285 def __sub__(self, other):
286 """
287 Overloading the subtraction operator for fields types
289 Args:
290 other (fields): fields object to be subtracted
291 Raises:
292 DataError: if other is not a fields object
293 Returns:
294 fields: differences between caller and other values (self-other)
295 """
297 if isinstance(other, type(self)):
298 # always create new fields, since otherwise c = a - b changes a as well!
299 p = fields(self)
300 p.elec[:] = self.elec - other.elec
301 p.magn[:] = self.magn - other.magn
302 return p
303 else:
304 raise DataError("Type error: cannot subtract %s from %s" % (type(other), type(self)))
306 def __rmul__(self, other):
307 """
308 Overloading the multiply with factor from right operator for fields types
310 Args:
311 other (float): factor
312 Raises:
313 DataError: if other is not a fields object
314 Returns:
315 fields: scaled fields
316 """
318 if isinstance(other, float):
319 # always create new fields, since otherwise c = a - b changes a as well!
320 p = fields(self)
321 p.elec[:] = other * self.elec
322 p.magn[:] = other * self.magn
323 return p
324 else:
325 raise DataError("Type error: cannot multiply %s with %s" % (type(other), type(self)))