Advertisement
Not a member of Pastebin yet?
Sign Up,
it unlocks many cool features!
- // Tomado y adaptado de http://zarb.org/~gc/html/libpng.html
- #include <unistd.h>
- #include <stdlib.h>
- #include <stdio.h>
- #include <string.h>
- #include <stdarg.h>
- #include <sys/time.h>
- #include <omp.h>
- #include <png.h>
- #include <sched.h>
- // For the CUDA runtime routines (prefixed with "cuda_")
- #include <cuda_runtime.h>
- void abort_(const char * s, ...)
- {
- va_list args;
- va_start(args, s);
- vfprintf(stderr, s, args);
- fprintf(stderr, "\n");
- va_end(args);
- abort();
- }
- int x, y;
- int width, height;
- int * d_width, *d_height;
- png_byte color_type;
- png_byte bit_depth;
- png_structp png_ptr;
- png_infop info_ptr;
- int number_of_passes;
- png_bytep * row_pointers;
- png_bytep * d_row_pointers;
- struct timeval tval_before, tval_after, tval_result, tval_before1, tval_after1, tval_result1;
- void read_png_file(char* file_name)
- {
- gettimeofday(&tval_before, NULL);
- char header[8]; // 8 is the maximum size that can be checked
- // Lee el archivo y verifica si es un PNG
- FILE *fp = fopen(file_name, "rb");
- if (!fp)
- abort_("[read_png_file] File %s could not be opened for reading", file_name);
- fread(header, 1, 8, fp);
- // if (png_sig_cmp(header, 0, 8))
- // abort_("[read_png_file] File %s is not recognized as a PNG file", file_name);
- //Inicializa variables necesarias para libpng
- png_ptr = png_create_read_struct (PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
- if (!png_ptr)
- abort_("[read_png_file] png_create_read_struct failed");
- info_ptr = png_create_info_struct(png_ptr);
- if (!info_ptr)
- abort_("[read_png_file] png_create_info_struct failed");
- if (setjmp(png_jmpbuf(png_ptr)))
- abort_("[read_png_file] Error during init_io");
- //Inicializa el input/output para el archivo PNG
- png_init_io(png_ptr, fp);
- png_set_sig_bytes(png_ptr, 8);
- //Lee la información anterior a los datos de los píxeles como tal
- png_read_info(png_ptr, info_ptr);
- //Almacena información del archivo PNG
- width = png_get_image_width(png_ptr, info_ptr);
- height = png_get_image_height(png_ptr, info_ptr);
- color_type = png_get_color_type(png_ptr, info_ptr);
- bit_depth = png_get_bit_depth(png_ptr, info_ptr);
- number_of_passes = png_set_interlace_handling(png_ptr);
- png_read_update_info(png_ptr, info_ptr);
- // Lectura del archivo PNG
- if (setjmp(png_jmpbuf(png_ptr)))
- abort_("[read_png_file] Error during read_image");
- // Reserva el espacio necesario para almacenar los datos del archivo PNG por filas
- row_pointers = (png_bytep*) malloc(sizeof(png_bytep) * height);
- for (y=0; y<height; y++)
- row_pointers[y] = (png_byte*) malloc(png_get_rowbytes(png_ptr,info_ptr));
- // Y para la copia para el device
- d_row_pointers = (png_bytep*) malloc(sizeof(png_bytep) * height);
- for (y=0; y<height; y++)
- d_row_pointers[y] = (png_byte*) malloc(png_get_rowbytes(png_ptr,info_ptr));
- png_read_image(png_ptr, row_pointers);
- fclose(fp);
- gettimeofday(&tval_after, NULL);
- timersub(&tval_after, &tval_before, &tval_result);
- printf("BnW Read: %ld.%06ld\n", (long int)tval_result.tv_sec, (long int)tval_result.tv_usec);
- }
- void write_png_file(char* file_name)
- {
- gettimeofday(&tval_before, NULL);
- // Crea el archivo
- FILE *fp = fopen(file_name, "wb");
- if (!fp)
- abort_("[write_png_file] File %s could not be opened for writing", file_name);
- //Inicializa variables necesarias para libpng
- png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
- if (!png_ptr)
- abort_("[write_png_file] png_create_write_struct failed");
- info_ptr = png_create_info_struct(png_ptr);
- if (!info_ptr)
- abort_("[write_png_file] png_create_info_struct failed");
- if (setjmp(png_jmpbuf(png_ptr)))
- abort_("[write_png_file] Error during init_io");
- png_init_io(png_ptr, fp);
- // Escribe el header del archivo PNG
- if (setjmp(png_jmpbuf(png_ptr)))
- abort_("[write_png_file] Error during writing header");
- png_set_IHDR(png_ptr, info_ptr, width, height,
- bit_depth, color_type, PNG_INTERLACE_NONE,
- PNG_COMPRESSION_TYPE_BASE, PNG_FILTER_TYPE_BASE);
- png_write_info(png_ptr, info_ptr);
- // Escribe los bytes del archivo PNG
- if (setjmp(png_jmpbuf(png_ptr)))
- abort_("[write_png_file] Error during writing bytes");
- png_write_image(png_ptr, row_pointers);
- // Termina la escritura
- if (setjmp(png_jmpbuf(png_ptr)))
- abort_("[write_png_file] Error during end of write");
- png_write_end(png_ptr, NULL);
- // Libera el espacio reservado previamente
- for (y=0; y<height; y++)
- free(row_pointers[y]);
- free(row_pointers);
- fclose(fp);
- gettimeofday(&tval_after, NULL);
- timersub(&tval_after, &tval_before, &tval_result);
- printf("BnW Write: %ld.%06ld\n", (long int)tval_result.tv_sec, (long int)tval_result.tv_usec);
- }
- __global__ void process_file(png_bytep * d_row_pointers, int * d_width, int * d_height)
- {
- // Se realizan los cambios deseados en la imagen
- //Verificar los datos recibidos
- printf("Width = %d , Height = %d ", *d_width, *d_height);
- int rgb_total = 0;
- float rgb_average = 0.0;
- int x = 0;
- int y = 0;
- png_byte *row;
- png_byte *ptr;
- for (y=0; y<*d_height-1; y++) {
- for (x=0; x<*d_width; x++) {
- rgb_total = 0;
- rgb_average = 0;
- row = d_row_pointers[y];
- ptr = &(row[x*3]);
- printf("Pixel %d - %d, Rgb values: %d - %d - %d \n", x, y, ptr[0], ptr[1], ptr[2]);
- rgb_total += ptr[0] + ptr[1] + ptr[2];
- // Calculando el promedios RGB
- rgb_average = rgb_total / 3;
- // printf("Average: %d \n", (int)rgb_average);
- ptr[0] = (int)rgb_average;
- ptr[1] = (int)rgb_average;
- ptr[2] = (int)rgb_average;
- // printf("Changed to %d - %d - %d \n",ptr[0], ptr[1], ptr[2]);
- // printf("Pixel %d - %d done\n",x,y);
- }
- }
- // printf("Para la imagen de resolución: %d x %d - ", *width, *height);
- }
- int main(int argc, char **argv)
- {
- png_byte *row;
- png_byte *ptr;
- // Verifica los parámetros para ejecutar el programa
- if (argc != 3)
- abort_("Uso: ./Nombre_del_Programa <file_in> <file_out>");
- // Timer de inicio total
- gettimeofday(&tval_before1, NULL);
- printf("-0");
- read_png_file(argv[1]);
- printf("-1");
- row = d_row_pointers[0];
- ptr = &(row[x*3]);
- printf("Pixel %d - %d, Rgb values: %d - %d - %d \n", x, y, ptr[0], ptr[1], ptr[2]);
- // CUDA
- // Tiempo de inicio para el procesamiento de la imagen
- gettimeofday(&tval_before, NULL);
- // Reservar el espacio para las copias de row_pointers, widht y height en el device
- int size = sizeof(png_bytep);
- int int_size = sizeof(int);
- printf("-2");
- cudaMalloc((void **)&d_row_pointers, size);
- // for (y=0; y<height; y++)
- // cudaMalloc((void **)&d_row_pointers[y],png_get_rowbytes(png_ptr,info_ptr));
- printf("-3");
- cudaMalloc((void **)&d_width, int_size);
- cudaMalloc((void **)&d_height, int_size);
- // Copiar los inputs al device
- cudaMemcpy(d_row_pointers, &row_pointers, size, cudaMemcpyHostToDevice);
- // for (y=0; y<height; y++)
- // cudaMemcpy(d_row_pointers[y], &row_pointers[y], png_get_rowbytes(png_ptr,info_ptr), cudaMemcpyHostToDevice);
- cudaMemcpy(d_width, &width, int_size, cudaMemcpyHostToDevice);
- cudaMemcpy(d_height, &height, int_size, cudaMemcpyHostToDevice);
- // Lanzar el kernel
- process_file<<<1,1>>>(d_row_pointers, d_width, d_height);
- // Copiar los resultados de vuelta al host
- cudaMemcpy(&row_pointers, d_row_pointers, size, cudaMemcpyDeviceToHost);
- // Limpieza
- for (y=0; y<height; y++)
- cudaFree(d_row_pointers[y]);
- cudaFree(d_row_pointers);
- cudaFree(d_width);
- cudaFree(d_height);
- // Tiempo de fin para el procesamiento de la imagen
- gettimeofday(&tval_after, NULL);
- timersub(&tval_after, &tval_before, &tval_result);
- printf("BnW Process: %ld.%06ld\n", (long int)tval_result.tv_sec, (long int)tval_result.tv_usec);
- // Escritura de la imagen con los resultados
- write_png_file(argv[2]);
- // Tiempo de fin total
- gettimeofday(&tval_after1, NULL);
- timersub(&tval_after1, &tval_before1, &tval_result1);
- printf("Tiempo de ejecución de efecto blanco y negro: %ld.%06ld\n \n", (long int)tval_result1.tv_sec, (long int)tval_result1.tv_usec);
- return 0;
- }
Advertisement
Advertisement