O tempo entre execuções de processos é ocasionalmente muito longo, como posso definir um tempo máximo? [fechadas]

2

Estou usando um pi de framboesa para capturar dados por meio de uma porta SPI. Anexado à porta SPI está um PIC, programado para que seja amostrado em um A / D a 12.8kHz e armazene as amostras em um buffer de 256 longos. (para lidar com o processo de linux pause e execute).

Eu tenho um programa Linux que lê o SPI e grava em um arquivo. um par de vezes por segundo estou recebendo dados em falta, acredito, porque o Linux está demorando muito para voltar ao meu programa em C.

Eu entendo que o Linux (Raspbian) não é em tempo real e um processo / programa pode ser interrompido a qualquer momento, e é por isso que estou usando um buffer no PIC, para passar por essas interrupções.

O problema é que, às vezes, o atraso em que o programa aguarda é muito grande e, portanto, estou perdendo alguns dos dados. Isso acontece talvez uma vez a cada poucos segundos:

Gostaria de sugestões para reduzir o tempo ou, pelo menos, definir um tempo máximo entre as fatias do tempo de execução do processo.

Eu tentei rodar com um bom nível de -15 e -20.

Existe uma maneira de reduzir as fatias de tempo?

Existe um disco de memória em Raspbian que eu posso tentar escrever meu arquivo, caso seja a escrita do arquivo que está causando isso?

Aqui estão alguns resultados da parte superior:

top - 14:01:23 up  2:25,  3 users,  load average: 1.91, 2.09, 2.18
Tasks:  76 total,   1 running,  75 sleeping,   0 stopped,   0 zombie
%Cpu(s):  7.6 us, 28.9 sy,  0.0 ni, 63.2 id,  0.0 wa,  0.0 hi,  0.3 si,  0.0 st
KiB Mem:    185732 total,   105708 used,    80024 free,    21112 buffers
KiB Swap:   102396 total,        0 used,   102396 free,    43916 cached

  PID USER      PR  NI  VIRT  RES  SHR S  %CPU %MEM    TIME+  COMMAND                                                  
  353 root      20   0     0    0    0 D  15.3  0.0  17:01.87 spi0                                                     
13296 root       5 -15  2652 1468 1344 D  12.1  0.8   0:09.21 a.out                                                    
13386 pi        20   0  4708 2500 2064 R   1.6  1.3   0:00.40 top                                                      
 4707 pi        20   0  4176 2572 1976 S   1.0  1.4   2:48.66 watch                                                    
11547 root      20   0     0    0    0 S   0.6  0.0   0:03.59 kworker/u2:1                                             
   16 root      20   0     0    0    0 S   0.3  0.0   0:05.60 kworker/0:1                                              
 2524 pi        20   0  9288 3296 2708 S   0.3  1.8   0:00.62 sshd                                                     
12448 root      20   0     0    0    0 S   0.3  0.0   0:02.98 kworker/u2:4                                             
    1 root      20   0  2168 1368 1260 S   0.0  0.7   0:02.22 init                                                     
    2 root      20   0     0    0    0 S   0.0  0.0   0:00.00 kthreadd                                                 
    3 root      20   0     0    0    0 S   0.0  0.0   0:06.88 ksoftirqd/0                                              
    5 root       0 -20     0    0    0 S   0.0  0.0   0:00.00  

Eu posso ver de cima que o processo a.out e spi0 estão usando o maior tempo de CPU, mas eles não estão maximizando isso. Aqui você pode ver que meu processo tem um bom de -15.

Talvez um dos processos em execução esteja interferindo? Aqui estão meus processos em execução, a.out é meu programa de leitura de SPI.

pi@raspberrypi ~/frd/src/raspi/xcodefrd $ ps -ae
  PID TTY          TIME CMD
    1 ?        00:00:02 init
    2 ?        00:00:00 kthreadd
    3 ?        00:00:07 ksoftirqd/0
    5 ?        00:00:00 kworker/0:0H
    7 ?        00:00:00 khelper
    8 ?        00:00:00 kdevtmpfs
    9 ?        00:00:00 netns
   10 ?        00:00:00 perf
   11 ?        00:00:00 khungtaskd
   12 ?        00:00:00 writeback
   13 ?        00:00:00 crypto
   14 ?        00:00:00 bioset
   15 ?        00:00:00 kblockd
   16 ?        00:00:06 kworker/0:1
   17 ?        00:00:00 rpciod
   18 ?        00:00:00 kswapd0
   19 ?        00:00:00 fsnotify_mark
   20 ?        00:00:00 nfsiod
   26 ?        00:00:00 kthrotld
   27 ?        00:00:00 VCHIQ-0
   28 ?        00:00:00 VCHIQr-0
   29 ?        00:00:00 VCHIQs-0
   30 ?        00:00:00 iscsi_eh
   31 ?        00:00:00 dwc_otg
   32 ?        00:00:00 DWC Notificatio
   34 ?        00:00:00 kworker/0:2
   35 ?        00:00:01 mmcqd/0
   36 ?        00:00:00 VCHIQka-0
   37 ?        00:00:00 SMIO
   38 ?        00:00:00 deferwq
   40 ?        00:00:00 jbd2/mmcblk0p6-
   41 ?        00:00:00 ext4-rsv-conver
  156 ?        00:00:00 udevd
  286 ?        00:00:00 udevd
  293 ?        00:00:00 udevd
  321 ?        00:00:00 cfg80211
  353 ?        00:18:14 spi0
  411 ?        00:00:00 kworker/0:1H
 1792 ?        00:00:06 ifplugd
 1795 ?        00:00:01 ifplugd
 1797 ?        00:00:01 ifplugd
 1800 ?        00:00:00 wpa_supplicant
 1877 ?        00:00:00 dhclient
 2246 ?        00:00:00 rsyslogd
 2248 ?        00:00:00 thd
 2296 ?        00:00:00 cron
 2326 ?        00:00:00 dbus-daemon
 2393 ?        00:00:01 ntpd
 2422 ?        00:00:00 sshd
 2513 tty1     00:00:00 getty
 2514 tty2     00:00:00 getty
 2515 tty3     00:00:00 getty
 2516 tty4     00:00:00 getty
 2517 tty5     00:00:00 getty
 2518 tty6     00:00:00 getty
 2519 ?        00:00:00 getty
 2520 ?        00:00:00 sshd
 2524 ?        00:00:04 sshd
 2525 pts/0    00:00:02 bash
 2545 ?        00:00:00 sshd
 2549 ?        00:00:39 sshd
 2550 ?        00:00:13 sftp-server
 2551 ?        00:00:00 sshd
 2555 ?        00:00:00 sshd
 2556 pts/1    00:00:01 bash
 2583 ?        00:00:00 sshd
 2587 ?        00:00:03 sshd
 2588 pts/2    00:00:01 bash
 4707 pts/2    00:02:56 watch
 8843 ?        00:00:06 kworker/u2:0
11547 ?        00:00:04 kworker/u2:1
12448 ?        00:00:03 kworker/u2:4
13186 ?        00:00:03 kworker/u2:2
13295 pts/0    00:00:00 sudo
13296 pts/0    00:01:09 a.out
13690 ?        00:00:01 kworker/u2:3
14083 pts/0    00:00:00 ps

É assim que meu fluxo de dados é exibido, você pode identificar o bit em que as leituras de SPI foram atrasadas:

Vocêpode,claro,quererdarumaolhadanomeucódigo.Eutenteidiferentestamanhosdebuffer.

#include<stdio.h>#include<stdlib.h>#include<wiringPi.h>#include<sys/timeb.h>#include<time.h>#include<sys/timeb.h>#include"frdSPI.h"
#include "frdBuffer.h"

#define TRIGGERVAL 50

#define BUFF_SIZE 32

int main(void){

    unsigned char buffer[BUFF_SIZE];
    int i, sample, channel = 0 , triggerdifference, oldsample, qtytowriteout = 0;
    unsigned long int samplenumber=0;
    unsigned long int differencenumber=0;

    FILE *fp;
    char filename[100];

    struct timeb recordtime;


    // Following is test code to see how to make filename and use it to write a file.
    ftime( &recordtime );
    sprintf( filename, "/frd/data/%ld%03ld.startup", recordtime.time, recordtime.millitm);

    fp = fopen( filename ,"w" );
    fprintf( fp, "test and more %lu", samplenumber);
    fclose(fp);



    if ( wiringPiSPISetup (channel, 500000) < 0){
        fprintf (stderr, "SPI Setup failed! Check module is loaded and run app as root.\n");
        exit(-1);
    }

    while(1){
        wiringPiSPIDataRW ( channel, &buffer, BUFF_SIZE );

        for(i=0; i<BUFF_SIZE; i++){
            sample=parseSPIDataStream(buffer[i]);
            if( sample > -1 ){
                samplenumber++;
                oldsample = circularbufferreadwrite( sample );

                if( calculateDifferenceValue(sample, &triggerdifference))
                {
                    differencenumber++;
                    if( triggerdifference > TRIGGERVAL || triggerdifference < (-1 * TRIGGERVAL) ){
                        //printf("%d \n", triggerdifference);
                        if( qtytowriteout == 0 )
                        {
                            ftime( &recordtime );
                            sprintf( filename, "/tmp/%ld%03ld.csv", recordtime.time, recordtime.millitm);
                            fp = fopen(filename,"w");
                            samplenumber=0;
                        }
                        if( qtytowriteout < 4000 )
                            qtytowriteout = qtytowriteout + 12800;
                    }
                }

                if( qtytowriteout && (fp!=(FILE *)NULL))
                {
                    //printf("%lu,%d\n", samplenumber, oldsample);
                    fprintf(fp, "%lu,%d\n", samplenumber, oldsample);
                    qtytowriteout--;
                    if( qtytowriteout == 0 ){
                        fclose(fp);
                        printf("\b");
                    }
                }
            }
        }
    }
}

Obrigado por qualquer ajuda e sugestões com antecedência.

EDIT Eu apenas corri dd if = / dev / zero de = / dev / null também, para ocupar algum tempo de CPU, e isso tornou os problemas significativamente piores, então eu acho que estou no caminho certo, mas eu poderia fazer com alguns ponteiros sobre como consertá-lo. Aqui está um gráfico dos dados ao usar mais tempo de CPU:

Um comentário abaixo consultou parseSPIDataStream, então incluímos aqui:

int parseSPIDataStream( char databyte ){

    static int state=0;
    static int result=0;
    static int ignorebytes=10;
    static int synced=FALSE;

    if( ignorebytes==0 )
    {

        switch( databyte )
        {
            case 0X7E:          //0x7E Filler and sync
                state=0;
                synced=TRUE;
                return(-1);

            case 0x7D:          //Next Char is escaped
                state=2;
                return(-1);

            default:
                break;
        }

        switch( state )
        {
            case 0:         //High Byte of result
                result = 256 * databyte;
                state=1;
                return(-1);

            case 1:         //Low byte of result
                result = result + databyte;
                state=0;
                if( synced )
                    return(result);
                else
                    return(-1);

            case 2:         //Escaped Low Byte
                result = result + 0x20 + databyte;
                state=0;
                if( synced )
                    return(result);
                else
                    return(-1);

            default:            //There is an error!!!
                return(-2);
        }

    }else{
        ignorebytes--;
        return(-3);
    }

}
    
por J Rodgers 28.07.2017 / 16:22

0 respostas