mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-25 13:17:41 +03:00 
			
		
		
		
	Don't try to read a multi-GB pg_stat_statements file in one call.
Windows fails on a request to read() more than INT_MAX bytes, and perhaps other platforms could have similar issues. Let's adjust this code to read at most 1GB per call. (One would not have thought the file could get that big, but now we have a field report of trouble, so it can. We likely ought to add some mechanism to limit the size of the query-texts file separately from the size of the hash table. That is not this patch, though.) Per bug #17254 from Yusuke Egashira. It's been like this for awhile, so back-patch to all supported branches. Discussion: https://postgr.es/m/17254-a926c89dc03375c2@postgresql.org
This commit is contained in:
		| @@ -1876,6 +1876,7 @@ qtext_load_file(Size *buffer_size) | |||||||
| 	char	   *buf; | 	char	   *buf; | ||||||
| 	int			fd; | 	int			fd; | ||||||
| 	struct stat stat; | 	struct stat stat; | ||||||
|  | 	Size		nread; | ||||||
|  |  | ||||||
| 	fd = OpenTransientFile(PGSS_TEXT_FILE, O_RDONLY | PG_BINARY, 0); | 	fd = OpenTransientFile(PGSS_TEXT_FILE, O_RDONLY | PG_BINARY, 0); | ||||||
| 	if (fd < 0) | 	if (fd < 0) | ||||||
| @@ -1916,28 +1917,40 @@ qtext_load_file(Size *buffer_size) | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	/* | 	/* | ||||||
| 	 * OK, slurp in the file.  If we get a short read and errno doesn't get | 	 * OK, slurp in the file.  Windows fails if we try to read more than | ||||||
| 	 * set, the reason is probably that garbage collection truncated the file | 	 * INT_MAX bytes at once, and other platforms might not like that either, | ||||||
| 	 * since we did the fstat(), so we don't log a complaint --- but we don't | 	 * so read a very large file in 1GB segments. | ||||||
| 	 * return the data, either, since it's most likely corrupt due to |  | ||||||
| 	 * concurrent writes from garbage collection. |  | ||||||
| 	 */ | 	 */ | ||||||
| 	errno = 0; | 	nread = 0; | ||||||
| 	if (read(fd, buf, stat.st_size) != stat.st_size) | 	while (nread < stat.st_size) | ||||||
| 	{ | 	{ | ||||||
| 		if (errno) | 		int			toread = Min(1024 * 1024 * 1024, stat.st_size - nread); | ||||||
| 			ereport(LOG, |  | ||||||
| 					(errcode_for_file_access(), | 		/* | ||||||
| 				   errmsg("could not read pg_stat_statement file \"%s\": %m", | 		 * If we get a short read and errno doesn't get set, the reason is | ||||||
| 						  PGSS_TEXT_FILE))); | 		 * probably that garbage collection truncated the file since we did | ||||||
| 		free(buf); | 		 * the fstat(), so we don't log a complaint --- but we don't return | ||||||
| 		CloseTransientFile(fd); | 		 * the data, either, since it's most likely corrupt due to concurrent | ||||||
| 		return NULL; | 		 * writes from garbage collection. | ||||||
|  | 		 */ | ||||||
|  | 		errno = 0; | ||||||
|  | 		if (read(fd, buf + nread, toread) != toread) | ||||||
|  | 		{ | ||||||
|  | 			if (errno) | ||||||
|  | 				ereport(LOG, | ||||||
|  | 						(errcode_for_file_access(), | ||||||
|  | 						 errmsg("could not read pg_stat_statement file \"%s\": %m", | ||||||
|  | 								PGSS_TEXT_FILE))); | ||||||
|  | 			free(buf); | ||||||
|  | 			CloseTransientFile(fd); | ||||||
|  | 			return NULL; | ||||||
|  | 		} | ||||||
|  | 		nread += toread; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	CloseTransientFile(fd); | 	CloseTransientFile(fd); | ||||||
|  |  | ||||||
| 	*buffer_size = stat.st_size; | 	*buffer_size = nread; | ||||||
| 	return buf; | 	return buf; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user