mirror of
				https://git.code.sf.net/p/libpng/code.git
				synced 2025-07-10 18:04:09 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			162 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			162 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* contrib/arm-neon/linux.c
 | |
|  *
 | |
|  * Copyright (c) 2014 Glenn Randers-Pehrson
 | |
|  * Written by John Bowler, 2014.
 | |
|  * Last changed in libpng 1.6.16 [December 22, 2014]
 | |
|  *
 | |
|  * This code is released under the libpng license.
 | |
|  * For conditions of distribution and use, see the disclaimer
 | |
|  * and license in png.h
 | |
|  *
 | |
|  * SEE contrib/arm-neon/README before reporting bugs
 | |
|  *
 | |
|  * STATUS: SUPPORTED
 | |
|  * BUG REPORTS: png-mng-implement@sourceforge.net
 | |
|  *
 | |
|  * png_have_neon implemented for Linux by reading the widely available
 | |
|  * pseudo-file /proc/cpuinfo.
 | |
|  *
 | |
|  * This code is strict ANSI-C and is probably moderately portable; it does
 | |
|  * however use <stdio.h> and it assumes that /proc/cpuinfo is never localized.
 | |
|  */
 | |
| #include <stdio.h>
 | |
| 
 | |
| static int
 | |
| png_have_neon(png_structp png_ptr)
 | |
| {
 | |
|    FILE *f = fopen("/proc/cpuinfo", "rb");
 | |
| 
 | |
|    if (f != NULL)
 | |
|    {
 | |
|       /* This is a simple state machine which reads the input byte-by-byte until
 | |
|        * it gets a match on the 'neon' feature or reaches the end of the stream.
 | |
|        */
 | |
|       static const char ch_feature[] = { 70, 69, 65, 84, 85, 82, 69, 83 };
 | |
|       static const char ch_neon[] = { 78, 69, 79, 78 };
 | |
| 
 | |
|       enum
 | |
|       {
 | |
|          StartLine, Feature, Colon, StartTag, Neon, HaveNeon, SkipTag, SkipLine
 | |
|       }  state;
 | |
|       int counter;
 | |
| 
 | |
|       for (state=StartLine, counter=0;;)
 | |
|       {
 | |
|          int ch = fgetc(f);
 | |
| 
 | |
|          if (ch == EOF)
 | |
|          {
 | |
|             /* EOF means error or end-of-file, return false; neon at EOF is
 | |
|              * assumed to be a mistake.
 | |
|              */
 | |
|             fclose(f);
 | |
|             return 0;
 | |
|          }
 | |
| 
 | |
|          switch (state)
 | |
|          {
 | |
|             case StartLine:
 | |
|                /* Match spaces at the start of line */
 | |
|                if (ch <= 32) /* skip control characters and space */
 | |
|                   break;
 | |
| 
 | |
|                counter=0;
 | |
|                state = Feature;
 | |
|                /* FALL THROUGH */
 | |
| 
 | |
|             case Feature:
 | |
|                /* Match 'FEATURE', ASCII case insensitive. */
 | |
|                if ((ch & ~0x20) == ch_feature[counter])
 | |
|                {
 | |
|                   if (++counter == (sizeof ch_feature))
 | |
|                      state = Colon;
 | |
|                   break;
 | |
|                }
 | |
|                
 | |
|                /* did not match 'feature' */
 | |
|                state = SkipLine;
 | |
|                /* FALL THROUGH */
 | |
| 
 | |
|             case SkipLine:
 | |
|             skipLine:
 | |
|                /* Skip everything until we see linefeed or carriage return */
 | |
|                if (ch != 10 && ch != 13)
 | |
|                   break;
 | |
| 
 | |
|                state = StartLine;
 | |
|                break;
 | |
| 
 | |
|             case Colon:
 | |
|                /* Match any number of space or tab followed by ':' */
 | |
|                if (ch == 32 || ch == 9)
 | |
|                   break;
 | |
| 
 | |
|                if (ch == 58) /* i.e. ':' */
 | |
|                {
 | |
|                   state = StartTag;
 | |
|                   break;
 | |
|                }
 | |
| 
 | |
|                /* Either a bad line format or a 'feature' prefix followed by
 | |
|                 * other characters.
 | |
|                 */
 | |
|                state = SkipLine;
 | |
|                goto skipLine;
 | |
| 
 | |
|             case StartTag:
 | |
|                /* Skip space characters before a tag */
 | |
|                if (ch == 32 || ch == 9)
 | |
|                   break;
 | |
| 
 | |
|                state = Neon;
 | |
|                counter = 0;
 | |
|                /* FALL THROUGH */
 | |
| 
 | |
|             case Neon:
 | |
|                /* Look for 'neon' tag */
 | |
|                if ((ch & ~0x20) == ch_neon[counter])
 | |
|                {
 | |
|                   if (++counter == (sizeof ch_neon))
 | |
|                      state = HaveNeon;
 | |
|                   break;
 | |
|                }
 | |
| 
 | |
|                state = SkipTag;
 | |
|                /* FALL THROUGH */
 | |
| 
 | |
|             case SkipTag:
 | |
|                /* Skip non-space characters */
 | |
|                if (ch == 10 || ch == 13)
 | |
|                   state = StartLine;
 | |
| 
 | |
|                else if (ch == 32 || ch == 9)
 | |
|                   state = StartTag;
 | |
|                break;
 | |
| 
 | |
|             case HaveNeon:
 | |
|                /* Have seen a 'neon' prefix, but there must be a space or new
 | |
|                 * line character to terminate it.
 | |
|                 */
 | |
|                if (ch == 10 || ch == 13 || ch == 32 || ch == 9)
 | |
|                {
 | |
|                   fclose(f);
 | |
|                   return 1;
 | |
|                }
 | |
| 
 | |
|                state = SkipTag;
 | |
|                break;
 | |
| 
 | |
|             default:
 | |
|                png_error(png_ptr, "png_have_neon: internal error (bug)");
 | |
|          }
 | |
|       }
 | |
|    }
 | |
| 
 | |
| #ifdef PNG_WARNINGS_SUPPORTED
 | |
|    else
 | |
|       png_warning(png_ptr, "/proc/cpuinfo open failed");
 | |
| #endif
 | |
| 
 | |
|    return 0;
 | |
| }
 | 
