Actual source code: reg.c
petsc-3.10.5 2019-03-28
2: /*
3: Provides a general mechanism to allow one to register new routines in
4: dynamic libraries for many of the PETSc objects (including, e.g., KSP and PC).
5: */
6: #include <petsc/private/petscimpl.h>
7: #include <petscviewer.h>
9: /*
10: This is the default list used by PETSc with the PetscDLLibrary register routines
11: */
12: PetscDLLibrary PetscDLLibrariesLoaded = 0;
14: #if defined(PETSC_HAVE_DYNAMIC_LIBRARIES) && defined(PETSC_USE_SHARED_LIBRARIES)
16: PetscErrorCode PetscLoadDynamicLibrary(const char *name,PetscBool *found)
17: {
18: char libs[PETSC_MAX_PATH_LEN],dlib[PETSC_MAX_PATH_LEN];
22: PetscStrncpy(libs,"${PETSC_LIB_DIR}/libpetsc",sizeof(libs));
23: PetscStrlcat(libs,name,sizeof(libs));
24: PetscDLLibraryRetrieve(PETSC_COMM_WORLD,libs,dlib,1024,found);
25: if (*found) {
26: PetscDLLibraryAppend(PETSC_COMM_WORLD,&PetscDLLibrariesLoaded,dlib);
27: } else {
28: PetscStrncpy(libs,"${PETSC_DIR}/${PETSC_ARCH}/lib/libpetsc",sizeof(libs));
29: PetscStrlcat(libs,name,sizeof(libs));
30: PetscDLLibraryRetrieve(PETSC_COMM_WORLD,libs,dlib,1024,found);
31: if (*found) {
32: PetscDLLibraryAppend(PETSC_COMM_WORLD,&PetscDLLibrariesLoaded,dlib);
33: }
34: }
35: return(0);
36: }
38: #endif
40: #if defined(PETSC_HAVE_THREADSAFETY)
41: PETSC_EXTERN PetscErrorCode AOInitializePackage(void);
42: PETSC_EXTERN PetscErrorCode PetscSFInitializePackage(void);
43: #if !defined(PETSC_USE_COMPLEX)
44: PETSC_EXTERN PetscErrorCode CharacteristicInitializePackage(void);
45: #endif
46: PETSC_EXTERN PetscErrorCode ISInitializePackage(void);
47: PETSC_EXTERN PetscErrorCode VecInitializePackage(void);
48: PETSC_EXTERN PetscErrorCode MatInitializePackage(void);
49: PETSC_EXTERN PetscErrorCode DMInitializePackage(void);
50: PETSC_EXTERN PetscErrorCode PCInitializePackage(void);
51: PETSC_EXTERN PetscErrorCode KSPInitializePackage(void);
52: PETSC_EXTERN PetscErrorCode SNESInitializePackage(void);
53: PETSC_EXTERN PetscErrorCode TSInitializePackage(void);
54: static MPI_Comm PETSC_COMM_WORLD_INNER = 0,PETSC_COMM_SELF_INNER = 0;
55: #endif
57: /*
58: PetscInitialize_DynamicLibraries - Adds the default dynamic link libraries to the
59: search path.
60: */
61: PETSC_INTERN PetscErrorCode PetscInitialize_DynamicLibraries(void)
62: {
63: char *libname[32];
65: PetscInt nmax,i;
66: #if defined(PETSC_USE_DYNAMIC_LIBRARIES) && defined(PETSC_USE_SHARED_LIBRARIES)
67: PetscBool preload;
68: #endif
71: nmax = 32;
72: PetscOptionsGetStringArray(NULL,NULL,"-dll_prepend",libname,&nmax,NULL);
73: for (i=0; i<nmax; i++) {
74: PetscDLLibraryPrepend(PETSC_COMM_WORLD,&PetscDLLibrariesLoaded,libname[i]);
75: PetscFree(libname[i]);
76: }
78: #if !defined(PETSC_USE_DYNAMIC_LIBRARIES) || !defined(PETSC_USE_SHARED_LIBRARIES)
79: /*
80: This just initializes the most basic PETSc stuff.
82: The classes, from PetscDraw to PetscTS, are initialized the first
83: time an XXCreate() is called.
84: */
85: PetscSysInitializePackage();
86: #else
87: preload = PETSC_FALSE;
88: PetscOptionsGetBool(NULL,NULL,"-dynamic_library_preload",&preload,NULL);
89: if (preload) {
90: PetscBool found;
91: #if defined(PETSC_USE_SINGLE_LIBRARY)
92: PetscLoadDynamicLibrary("",&found);
93: if (!found) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to locate PETSc dynamic library \n You cannot move the dynamic libraries!");
94: #else
95: PetscLoadDynamicLibrary("sys",&found);
96: if (!found) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to locate PETSc dynamic library \n You cannot move the dynamic libraries!");
97: PetscLoadDynamicLibrary("vec",&found);
98: if (!found) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to locate PETSc Vec dynamic library \n You cannot move the dynamic libraries!");
99: PetscLoadDynamicLibrary("mat",&found);
100: if (!found) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to locate PETSc Mat dynamic library \n You cannot move the dynamic libraries!");
101: PetscLoadDynamicLibrary("dm",&found);
102: if (!found) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to locate PETSc DM dynamic library \n You cannot move the dynamic libraries!");
103: PetscLoadDynamicLibrary("ksp",&found);
104: if (!found) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to locate PETSc KSP dynamic library \n You cannot move the dynamic libraries!");
105: PetscLoadDynamicLibrary("snes",&found);
106: if (!found) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to locate PETSc SNES dynamic library \n You cannot move the dynamic libraries!");
107: PetscLoadDynamicLibrary("ts",&found);
108: if (!found) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to locate PETSc TS dynamic library \n You cannot move the dynamic libraries!");
109: #endif
110: }
111: #endif
113: nmax = 32;
114: PetscOptionsGetStringArray(NULL,NULL,"-dll_append",libname,&nmax,NULL);
115: for (i=0; i<nmax; i++) {
116: PetscDLLibraryAppend(PETSC_COMM_WORLD,&PetscDLLibrariesLoaded,libname[i]);
117: PetscFree(libname[i]);
118: }
120: #if defined(PETSC_HAVE_THREADSAFETY)
121: /* These must be done here because it is not safe for individual threads to call these initialize routines */
122: AOInitializePackage();
123: PetscSFInitializePackage();
124: #if !defined(PETSC_USE_COMPLEX)
125: CharacteristicInitializePackage();
126: #endif
127: ISInitializePackage();
128: VecInitializePackage();
129: MatInitializePackage();
130: DMInitializePackage();
131: PCInitializePackage();
132: KSPInitializePackage();
133: SNESInitializePackage();
134: TSInitializePackage();
135: PetscCommDuplicate(PETSC_COMM_SELF,&PETSC_COMM_SELF_INNER,NULL);
136: PetscCommDuplicate(PETSC_COMM_WORLD,&PETSC_COMM_WORLD_INNER,NULL);
137: #endif
138: return(0);
139: }
141: /*
142: PetscFinalize_DynamicLibraries - Closes the opened dynamic libraries.
143: */
144: PETSC_INTERN PetscErrorCode PetscFinalize_DynamicLibraries(void)
145: {
147: PetscBool flg = PETSC_FALSE;
150: PetscOptionsGetBool(NULL,NULL,"-dll_view",&flg,NULL);
151: if (flg) { PetscDLLibraryPrintPath(PetscDLLibrariesLoaded); }
152: PetscDLLibraryClose(PetscDLLibrariesLoaded);
154: #if defined(PETSC_HAVE_THREADSAFETY)
155: PetscCommDestroy(&PETSC_COMM_SELF_INNER);
156: PetscCommDestroy(&PETSC_COMM_WORLD_INNER);
157: #endif
159: PetscDLLibrariesLoaded = 0;
160: return(0);
161: }
165: /* ------------------------------------------------------------------------------*/
166: struct _n_PetscFunctionList {
167: void (*routine)(void); /* the routine */
168: char *name; /* string to identify routine */
169: PetscFunctionList next; /* next pointer */
170: PetscFunctionList next_list; /* used to maintain list of all lists for freeing */
171: };
173: /*
174: Keep a linked list of PetscFunctionLists so that we can destroy all the left-over ones.
175: */
176: static PetscFunctionList dlallhead = 0;
178: /*MC
179: PetscFunctionListAdd - Given a routine and a string id, saves that routine in the
180: specified registry.
182: Synopsis:
183: #include <petscsys.h>
184: PetscErrorCode PetscFunctionListAdd(PetscFunctionList *flist,const char name[],void (*fptr)(void))
186: Not Collective
188: Input Parameters:
189: + flist - pointer to function list object
190: . name - string to identify routine
191: - fptr - function pointer
193: Notes:
194: To remove a registered routine, pass in a NULL fptr.
196: Users who wish to register new classes for use by a particular PETSc
197: component (e.g., SNES) should generally call the registration routine
198: for that particular component (e.g., SNESRegister()) instead of
199: calling PetscFunctionListAdd() directly.
201: Level: developer
203: .seealso: PetscFunctionListDestroy(), SNESRegister(), KSPRegister(),
204: PCRegister(), TSRegister(), PetscFunctionList, PetscObjectComposeFunction()
205: M*/
206: PETSC_EXTERN PetscErrorCode PetscFunctionListAdd_Private(PetscFunctionList *fl,const char name[],void (*fnc)(void))
207: {
208: PetscFunctionList entry,ne;
209: PetscErrorCode ierr;
212: if (!*fl) {
213: PetscNew(&entry);
214: PetscStrallocpy(name,&entry->name);
215: entry->routine = fnc;
216: entry->next = 0;
217: *fl = entry;
219: #if defined(PETSC_USE_DEBUG)
220: /* add this new list to list of all lists */
221: if (!dlallhead) {
222: dlallhead = *fl;
223: (*fl)->next_list = 0;
224: } else {
225: ne = dlallhead;
226: dlallhead = *fl;
227: (*fl)->next_list = ne;
228: }
229: #endif
231: } else {
232: /* search list to see if it is already there */
233: ne = *fl;
234: while (ne) {
235: PetscBool founddup;
237: PetscStrcmp(ne->name,name,&founddup);
238: if (founddup) { /* found duplicate */
239: ne->routine = fnc;
240: return(0);
241: }
242: if (ne->next) ne = ne->next;
243: else break;
244: }
245: /* create new entry and add to end of list */
246: PetscNew(&entry);
247: PetscStrallocpy(name,&entry->name);
248: entry->routine = fnc;
249: entry->next = 0;
250: ne->next = entry;
251: }
252: return(0);
253: }
255: /*@
256: PetscFunctionListDestroy - Destroys a list of registered routines.
258: Input Parameter:
259: . fl - pointer to list
261: Level: developer
263: .seealso: PetscFunctionListAdd(), PetscFunctionList
264: @*/
265: PetscErrorCode PetscFunctionListDestroy(PetscFunctionList *fl)
266: {
267: PetscFunctionList next,entry,tmp = dlallhead;
268: PetscErrorCode ierr;
271: if (!*fl) return(0);
273: /*
274: Remove this entry from the master DL list (if it is in it)
275: */
276: if (dlallhead == *fl) {
277: if (dlallhead->next_list) dlallhead = dlallhead->next_list;
278: else dlallhead = NULL;
279: } else if (tmp) {
280: while (tmp->next_list != *fl) {
281: tmp = tmp->next_list;
282: if (!tmp->next_list) break;
283: }
284: if (tmp->next_list) tmp->next_list = tmp->next_list->next_list;
285: }
287: /* free this list */
288: entry = *fl;
289: while (entry) {
290: next = entry->next;
291: PetscFree(entry->name);
292: PetscFree(entry);
293: entry = next;
294: }
295: *fl = 0;
296: return(0);
297: }
299: /*
300: Print any PetscFunctionLists that have not be destroyed
301: */
302: PetscErrorCode PetscFunctionListPrintAll(void)
303: {
304: PetscFunctionList tmp = dlallhead;
305: PetscErrorCode ierr;
308: if (tmp) {
309: PetscPrintf(PETSC_COMM_WORLD,"The following PetscFunctionLists were not destroyed\n");
310: }
311: while (tmp) {
312: PetscPrintf(PETSC_COMM_WORLD,"%s \n",tmp->name);
313: tmp = tmp->next_list;
314: }
315: return(0);
316: }
318: /*MC
319: PetscFunctionListFind - Find function registered under given name
321: Synopsis:
322: #include <petscsys.h>
323: PetscErrorCode PetscFunctionListFind(PetscFunctionList flist,const char name[],void (**fptr)(void))
325: Input Parameters:
326: + flist - pointer to list
327: - name - name registered for the function
329: Output Parameters:
330: . fptr - the function pointer if name was found, else NULL
332: Level: developer
334: .seealso: PetscFunctionListAdd(), PetscFunctionList, PetscObjectQueryFunction()
335: M*/
336: PETSC_EXTERN PetscErrorCode PetscFunctionListFind_Private(PetscFunctionList fl,const char name[],void (**r)(void))
337: {
338: PetscFunctionList entry = fl;
339: PetscErrorCode ierr;
340: PetscBool flg;
343: if (!name) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_ARG_NULL,"Trying to find routine with null name");
345: *r = 0;
346: while (entry) {
347: PetscStrcmp(name,entry->name,&flg);
348: if (flg) {
349: *r = entry->routine;
350: return(0);
351: }
352: entry = entry->next;
353: }
354: return(0);
355: }
357: /*@
358: PetscFunctionListView - prints out contents of an PetscFunctionList
360: Collective over MPI_Comm
362: Input Parameters:
363: + list - the list of functions
364: - viewer - currently ignored
366: Level: developer
368: .seealso: PetscFunctionListAdd(), PetscFunctionListPrintTypes(), PetscFunctionList
369: @*/
370: PetscErrorCode PetscFunctionListView(PetscFunctionList list,PetscViewer viewer)
371: {
373: PetscBool iascii;
376: if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
380: PetscObjectTypeCompare((PetscObject)viewer,PETSCVIEWERASCII,&iascii);
381: if (!iascii) SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SUP,"Only ASCII viewer supported");
383: while (list) {
384: PetscViewerASCIIPrintf(viewer," %s\n",list->name);
385: list = list->next;
386: }
387: PetscViewerASCIIPrintf(viewer,"\n");
388: return(0);
389: }
391: /*@C
392: PetscFunctionListGet - Gets an array the contains the entries in PetscFunctionList, this is used
393: by help etc.
395: Not Collective
397: Input Parameter:
398: . list - list of types
400: Output Parameter:
401: + array - array of names
402: - n - length of array
404: Notes:
405: This allocates the array so that must be freed. BUT the individual entries are
406: not copied so should not be freed.
408: Level: developer
410: .seealso: PetscFunctionListAdd(), PetscFunctionList
411: @*/
412: PetscErrorCode PetscFunctionListGet(PetscFunctionList list,const char ***array,int *n)
413: {
414: PetscErrorCode ierr;
415: PetscInt count = 0;
416: PetscFunctionList klist = list;
419: while (list) {
420: list = list->next;
421: count++;
422: }
423: PetscMalloc1(count+1,(char***)array);
424: count = 0;
425: while (klist) {
426: (*array)[count] = klist->name;
427: klist = klist->next;
428: count++;
429: }
430: (*array)[count] = 0;
431: *n = count+1;
432: return(0);
433: }
436: /*@C
437: PetscFunctionListPrintTypes - Prints the methods available.
439: Collective over MPI_Comm
441: Input Parameters:
442: + comm - the communicator (usually MPI_COMM_WORLD)
443: . fd - file to print to, usually stdout
444: . prefix - prefix to prepend to name (optional)
445: . name - option string (for example, "-ksp_type")
446: . text - short description of the object (for example, "Krylov solvers")
447: . man - name of manual page that discusses the object (for example, "KSPCreate")
448: . list - list of types
449: - def - default (current) value
451: Level: developer
453: .seealso: PetscFunctionListAdd(), PetscFunctionList
454: @*/
455: PetscErrorCode PetscFunctionListPrintTypes(MPI_Comm comm,FILE *fd,const char prefix[],const char name[],const char text[],const char man[],PetscFunctionList list,const char def[])
456: {
458: char p[64];
461: if (!fd) fd = PETSC_STDOUT;
463: PetscStrncpy(p,"-",sizeof(p));
464: if (prefix) {PetscStrlcat(p,prefix,sizeof(p));}
465: PetscFPrintf(comm,fd," %s%s <%s>: %s (one of)",p,name+1,def,text);
467: while (list) {
468: PetscFPrintf(comm,fd," %s",list->name);
469: list = list->next;
470: }
471: PetscFPrintf(comm,fd," (%s)\n",man);
472: return(0);
473: }
475: /*@
476: PetscFunctionListDuplicate - Creates a new list from a given object list.
478: Input Parameters:
479: . fl - pointer to list
481: Output Parameters:
482: . nl - the new list (should point to 0 to start, otherwise appends)
484: Level: developer
486: .seealso: PetscFunctionList, PetscFunctionListAdd(), PetscFlistDestroy()
488: @*/
489: PetscErrorCode PetscFunctionListDuplicate(PetscFunctionList fl,PetscFunctionList *nl)
490: {
494: while (fl) {
495: PetscFunctionListAdd(nl,fl->name,fl->routine);
496: fl = fl->next;
497: }
498: return(0);
499: }