mirror of
				https://gitlab.gnome.org/GNOME/glib.git
				synced 2025-10-26 14:02:17 +01:00 
			
		
		
		
	gdate: add internal helper function _g_localtime() for localtime()/localtime_r()
The point of to be able to call localtime()/localtime_r() from another place inside glib (without reimplementing the #ifdef). - also handles failures from localtime_r(). It is documented that it might fail, so detect the failure. - in case of failures of localtime(), still initialize the GDate. Previously, we may error out with a g_critical() assertion before. However, now that failures from localtime_r() are also caught, I think we should make an effort to initialize the GDate to something. It either way it not supposed to happen.
This commit is contained in:
		
							
								
								
									
										69
									
								
								glib/gdate.c
									
									
									
									
									
								
							
							
						
						
									
										69
									
								
								glib/gdate.c
									
									
									
									
									
								
							| @@ -53,6 +53,7 @@ | |||||||
| #include "gtestutils.h" | #include "gtestutils.h" | ||||||
| #include "gthread.h" | #include "gthread.h" | ||||||
| #include "gunicode.h" | #include "gunicode.h" | ||||||
|  | #include "gutilsprivate.h" | ||||||
|  |  | ||||||
| #ifdef G_OS_WIN32 | #ifdef G_OS_WIN32 | ||||||
| #include "garray.h" | #include "garray.h" | ||||||
| @@ -1403,6 +1404,33 @@ g_date_set_parse (GDate       *d, | |||||||
|   G_UNLOCK (g_date_global); |   G_UNLOCK (g_date_global); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | gboolean | ||||||
|  | _g_localtime (time_t timet, struct tm *out_tm) | ||||||
|  | { | ||||||
|  |   gboolean success = TRUE; | ||||||
|  |  | ||||||
|  | #ifdef HAVE_LOCALTIME_R | ||||||
|  |   if (!localtime_r (&timet, out_tm)) | ||||||
|  |     success = FALSE; | ||||||
|  | #else | ||||||
|  |   { | ||||||
|  |     struct tm *ptm = localtime (&timet); | ||||||
|  |  | ||||||
|  |     if (ptm == NULL) | ||||||
|  |       { | ||||||
|  |         /* Happens at least in Microsoft's C library if you pass a | ||||||
|  |          * negative time_t. | ||||||
|  |          */ | ||||||
|  |         success = FALSE; | ||||||
|  |       } | ||||||
|  |     else | ||||||
|  |       memcpy (out_tm, ptm, sizeof (struct tm)); | ||||||
|  |   } | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  |   return success; | ||||||
|  | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * g_date_set_time_t: |  * g_date_set_time_t: | ||||||
|  * @date: a #GDate  |  * @date: a #GDate  | ||||||
| @@ -1427,33 +1455,21 @@ g_date_set_time_t (GDate *date, | |||||||
| 		   time_t timet) | 		   time_t timet) | ||||||
| { | { | ||||||
|   struct tm tm; |   struct tm tm; | ||||||
|    |   gboolean success; | ||||||
|  |  | ||||||
|   g_return_if_fail (date != NULL); |   g_return_if_fail (date != NULL); | ||||||
|    |  | ||||||
| #ifdef HAVE_LOCALTIME_R |  | ||||||
|   localtime_r (&timet, &tm); |  | ||||||
| #else |  | ||||||
|   { |  | ||||||
|     struct tm *ptm = localtime (&timet); |  | ||||||
|  |  | ||||||
|     if (ptm == NULL) |   success = _g_localtime (timet, &tm); | ||||||
|       { |   if (!success) | ||||||
| 	/* Happens at least in Microsoft's C library if you pass a |     { | ||||||
| 	 * negative time_t. Use 2000-01-01 as default date. |       /* Still set a default date, 2000-01-01. | ||||||
| 	 */ |        * | ||||||
| #ifndef G_DISABLE_CHECKS |        * We may assert out below. */ | ||||||
| 	g_return_if_fail_warning (G_LOG_DOMAIN, "g_date_set_time", "ptm != NULL"); |       tm.tm_mon = 0; | ||||||
| #endif |       tm.tm_mday = 1; | ||||||
|  |       tm.tm_year = 100; | ||||||
|  |     } | ||||||
|  |  | ||||||
| 	tm.tm_mon = 0; |  | ||||||
| 	tm.tm_mday = 1; |  | ||||||
| 	tm.tm_year = 100; |  | ||||||
|       } |  | ||||||
|     else |  | ||||||
|       memcpy ((void *) &tm, (void *) ptm, sizeof(struct tm)); |  | ||||||
|   } |  | ||||||
| #endif |  | ||||||
|    |  | ||||||
|   date->julian = FALSE; |   date->julian = FALSE; | ||||||
|    |    | ||||||
|   date->month = tm.tm_mon + 1; |   date->month = tm.tm_mon + 1; | ||||||
| @@ -1463,6 +1479,11 @@ g_date_set_time_t (GDate *date, | |||||||
|   g_return_if_fail (g_date_valid_dmy (date->day, date->month, date->year)); |   g_return_if_fail (g_date_valid_dmy (date->day, date->month, date->year)); | ||||||
|    |    | ||||||
|   date->dmy    = TRUE; |   date->dmy    = TRUE; | ||||||
|  |  | ||||||
|  | #ifndef G_DISABLE_CHECKS | ||||||
|  |   if (!success) | ||||||
|  |     g_return_if_fail_warning (G_LOG_DOMAIN, "g_date_set_time", "localtime() == NULL"); | ||||||
|  | #endif | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -26,6 +26,8 @@ | |||||||
| #include "gtypes.h" | #include "gtypes.h" | ||||||
| #include "gtestutils.h" | #include "gtestutils.h" | ||||||
|  |  | ||||||
|  | #include <time.h> | ||||||
|  |  | ||||||
| G_BEGIN_DECLS | G_BEGIN_DECLS | ||||||
|  |  | ||||||
| void g_set_user_dirs (const gchar *first_dir_type, | void g_set_user_dirs (const gchar *first_dir_type, | ||||||
| @@ -55,6 +57,8 @@ g_nearest_pow (gsize num) | |||||||
|  |  | ||||||
| void _g_unset_cached_tmp_dir (void); | void _g_unset_cached_tmp_dir (void); | ||||||
|  |  | ||||||
|  | gboolean _g_localtime (time_t timet, struct tm *tm); | ||||||
|  |  | ||||||
| G_END_DECLS | G_END_DECLS | ||||||
|  |  | ||||||
| #endif /* __G_UTILS_PRIVATE_H__ */ | #endif /* __G_UTILS_PRIVATE_H__ */ | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user