Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Tastiera gaming MSI GK600 TKL: switch hot-swap, display LCD e tre modalità wireless
Tastiera gaming MSI GK600 TKL: switch hot-swap, display LCD e tre modalità wireless
MSI FORGE GK600 TKL WIRELESS: switch lineari hot-swap, tripla connettività, display LCD e 5 strati di fonoassorbimento. Ottima in gaming, a 79,99 euro
DJI Osmo Pocket 4: la gimbal camera tascabile cresce e ha nuovi controlli fisici
DJI Osmo Pocket 4: la gimbal camera tascabile cresce e ha nuovi controlli fisici
DJI porta un importante aggiornamento alla sua linea di gimbal camera tascabili con Osmo Pocket 4: sensore CMOS da 1 pollice rinnovato, gamma dinamica a 14 stop, profilo colore D-Log a 10 bit, slow motion a 4K/240fps e 107 GB di archiviazione integrata. Un prodotto pensato per i creator avanzati, ma che convince anche per l'uso quotidiano
Sony INZONE H6 Air: il primo headset open-back di Sony per giocatori
Sony INZONE H6 Air: il primo headset open-back di Sony per giocatori
Il primo headset open-back della linea INZONE arriva a 200 euro con driver derivati dalle cuffie da studio MDR-MV1 e un peso record di soli 199 grammi
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 01-03-2015, 01:15   #1
Vincenzoflaminio
Member
 
Iscritto dal: Jul 2006
Messaggi: 96
[C] trasfermento file Client/Server

Come da titolo non riesco ad effettuare in modo corretto un trasferimento di un file in particolare il file che si copia dal client al server ha una serie di NUL character (/0) che sembrano tra l'altro apparire in modo random e che non sono presenti nel file di origine.
Aggiungo che devo usare Read e Write e non sendfile.

Codice Client
Codice:
void Trasferisci(int DescrittoreSocket) {
    
    int  fd_sorg,nread;
    struct timeval tv;
	long int tempoTrasferimento;
	char buff[2048];
    DIR *dir;  // puntatore alla directory su cui lavorare
    struct dirent *file_dir; // struttura rappresentate i file della directory
    struct stat info_file; // struttura contenente info sul file
    char *nome_file; // stringa contenente il nome del file

	   gettimeofday(&tv, 0);
	   tempoTrasferimento = tv.tv_sec * 1000000 + tv.tv_usec;

       printf("Client: invio file\n");

		printf("Client: file inviato\n");
		gettimeofday(&tv, 0);
		tempoTrasferimento = (tv.tv_sec * 1000000 + tv.tv_usec)
				- tempoTrasferimento;

		printf("Trasferimento terminato, tempo di trasferimento microsec. %ld, \n",
				tempoTrasferimento);

    // mi sposto nella directory locale dello studente
    chdir(directory);  
    
    // apro la directory specificata passata
    if ((dir = opendir(directory)) == NULL)
        {
            perror("Errore apertura directory!!!");
            exit(-2);
        }

    // scorre la directory appena aperta
    while ((file_dir = readdir(dir)) != NULL) { 
    
        // recupero info sul file specificato da path
        if ((stat(file_dir->d_name, &info_file)) < 0)
              {
                perror("Errore stat!!!");
                exit(-3);
              }
        nome_file = file_dir->d_name; 
        int len =strlen(nome_file);
        nome_file[len]='\0';
 	    
	    // Apro il file per il trasferimento
		if ((fd_sorg = open(nome_file, O_RDONLY)) < 0)
		{
			perror("open file sorgente");
		}
		
		// trasferisco il file
		while ((nread = read(fd_sorg, buff, sizeof(buff)) > 0))
		{
			write(DescrittoreSocket, &buff, nread);
		}
              
    }
    
    // chiude il file
    close(fd_sorg);
        
    // chiude la directory appena utilizzata 
        closedir(dir);

return;
}
Codice Server
Codice:
int CopiaFile(int connfd,char path[]) {
   
   char info[2048];
   int fd_dest;
   DIR *dir;  // puntatore alla directory su cui lavorare
   char nome_file[80]; // Stringa contenente il nome file inviato dal client
   
   memset((char *) &nome_file, 0, sizeof(nome_file));
   memset((char *) &info, 0, sizeof(info)); 
   
   //leggo il nome del file
   read(connfd,&nome_file,sizeof(nome_file)); 
   int len=strlen(nome_file);
   nome_file[len] ='\0';
   printf("Il nome del file da trasferire e' %s di lunghezza %d\n",nome_file,len);

   // mi sposto nella directory remota
   chdir(path); 
  
   // apro la directory dello studente
   if ((dir = opendir(path)) == NULL)
   {
       perror("Errore apertura directory!!!");
       exit(-1);
   }
   fd_dest = open(nome_file,O_WRONLY|O_CREAT|O_APPEND,0777);//O_CREAT || O_TRUNC || O_WRONLY);
   if (fd_dest< 0) {
			perror("open file sorgente");
			exit(-1);
      }

   int letti,scritti;

//   while ((letti = read(connfd, &info, sizeof(info)) > 0) ) */
   while (1)
   			{
				
				if((letti=read(connfd, &info, sizeof(info))==0)) {
				     break;
				}     
				printf("info %s\n" , info);    
				printf("info %d\n" , letti);  
				scritti=write(fd_dest, &info, sizeof(info));
				printf("scritti =%d \n",scritti);
				break;
			}
 
  close(fd_dest);

  // chiude la directory appena utilizzata 
  closedir(dir);
 
return 0;

}
Vincenzoflaminio è offline   Rispondi citando il messaggio o parte di esso
Old 01-03-2015, 23:56   #2
lishi
Senior Member
 
L'Avatar di lishi
 
Iscritto dal: Sep 2006
Messaggi: 1539
Cosi su due piedi.

1) Manca da qualche parte la spedizione della lunghezza del file.
Ovvero

Prima del file devi spedire la lunghezza(lunghezza in byte fissa, 4 byte per un int dovrebbero bastare). A questo punto leggi i primi quattro byte dal socket e la condizione di stop del while sarà la lettura di un numero di byte pari alla lunghezza spedita. La condizione di leggere 0 byte non basta in quando per motivi di rete potresti leggere 0 byte anche se ci sono ancora dati da leggere.

Questo è un consiglio generale, non penso che possa dare il comportamento che tu vedi.

Per cui

2)
Leggi sempre la documentazione delle funzioni, in particolare se ti danno problemi.
read() può anche tornare un numero negativo, che indica un errore.
Se succedo questo il tuo programma suppone che i dati letti siano validi quando in realtà non lo sono.
lishi è offline   Rispondi citando il messaggio o parte di esso
Old 03-03-2015, 22:13   #3
Vincenzoflaminio
Member
 
Iscritto dal: Jul 2006
Messaggi: 96
Quote:
Originariamente inviato da lishi Guarda i messaggi
Cosi su due piedi.

1) Manca da qualche parte la spedizione della lunghezza del file.
Ovvero

Prima del file devi spedire la lunghezza(lunghezza in byte fissa, 4 byte per un int dovrebbero bastare). A questo punto leggi i primi quattro byte dal socket e la condizione di stop del while sarà la lettura di un numero di byte pari alla lunghezza spedita. La condizione di leggere 0 byte non basta in quando per motivi di rete potresti leggere 0 byte anche se ci sono ancora dati da leggere.

Questo è un consiglio generale, non penso che possa dare il comportamento che tu vedi.

Per cui

2)
Leggi sempre la documentazione delle funzioni, in particolare se ti danno problemi.
read() può anche tornare un numero negativo, che indica un errore.
Se succedo questo il tuo programma suppone che i dati letti siano validi quando in realtà non lo sono.
Sono uno studente quindi un principiante, ti ringrazio per la dritta ma non so come attuarla , come faccio a dire alla funzione quanti byte spedire?
La Read e la Write leggono COUNT byte dal Buffer(void * buf) è qui che sbaglio?
Vincenzoflaminio è offline   Rispondi citando il messaggio o parte di esso
Old 04-03-2015, 15:36   #4
Oceans11
Senior Member
 
L'Avatar di Oceans11
 
Iscritto dal: Sep 2005
Città: Torino
Messaggi: 606
Codice:
$ man 2 read
DESCRIPTION
       read()  attempts to read up to count bytes from file descriptor fd into
       the buffer starting at buf. [...]
RETURN VALUE
       On success, the number of bytes read is returned (zero indicates end of
       file), and the file position is advanced by this number.  It is not  an
       error  if  this  number  is smaller than the number of bytes requested;  
       this may happen for example because fewer bytes are actually  available
       right  now. [...]
Significa che devi sempre controllare se hai letto quanti byte hai richiesto di leggere e, in caso negativo, rileggere nuovamente.

Inoltre, per spedire un intero su socket devi stare attento alle dimensioni dell'intero (in byte, che sono machine e compiler dependent) e endianess.
(man ntohl e man htonl)
__________________
"Se proprio dovete piratare un prodotto, preferiamo che sia il nostro piuttosto che quello di qualcun altro." [Jeff Raikes]
"Pirating software? Choose Microsoft!"
Oceans11 è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


Tastiera gaming MSI GK600 TKL: switch hot-swap, display LCD e tre modalità wireless Tastiera gaming MSI GK600 TKL: switch hot-swap, ...
DJI Osmo Pocket 4: la gimbal camera tascabile cresce e ha nuovi controlli fisici DJI Osmo Pocket 4: la gimbal camera tascabile cr...
Sony INZONE H6 Air: il primo headset open-back di Sony per giocatori Sony INZONE H6 Air: il primo headset open-back d...
Nutanix cambia pelle: dall’iperconvergenza alla piattaforma full stack per cloud ibrido e IA Nutanix cambia pelle: dall’iperconvergenza alla ...
Recensione Xiaomi Pad 8 Pro: potenza bruta e HyperOS 3 per sfidare la fascia alta Recensione Xiaomi Pad 8 Pro: potenza bruta e Hyp...
iPhone 18 Pro: il componente che garanti...
DeepL alza il livello: con Voice-to-Voic...
Apple sta utilizzando sempre più ...
Il MacBook Neo vende tanto? Microsoft le...
AST SpaceMobile BlueBird 7: Blue Origin ...
È il momento migliore per comprar...
Svendita MacBook Pro: c'è il mode...
Oggi questa TV TCL QLED da 43 pollici co...
Il caricatore multiplo da 200W che va be...
Top 7 Amazon, il meglio del meglio di qu...
Spento lo strumento LECP della sonda spa...
Voyager Technologies ha siglato un accor...
GoPro annuncia la linea MISSION 1 con tr...
Alcune varianti dei futuri Samsung Galax...
Il ridimensionamento di OnePlus in Europ...
Chromium
GPU-Z
OCCT
LibreOffice Portable
Opera One Portable
Opera One 106
CCleaner Portable
CCleaner Standard
Cpu-Z
Driver NVIDIA GeForce 546.65 WHQL
SmartFTP
Trillian
Google Chrome Portable
Google Chrome 120
VirtualBox
Tutti gli articoli Tutte le news Tutti i download

Strumenti

Regole
Non Puoi aprire nuove discussioni
Non Puoi rispondere ai messaggi
Non Puoi allegare file
Non Puoi modificare i tuoi messaggi

Il codice vB è On
Le Faccine sono On
Il codice [IMG] è On
Il codice HTML è Off
Vai al Forum


Tutti gli orari sono GMT +1. Ora sono le: 04:28.


Powered by vBulletin® Version 3.6.4
Copyright ©2000 - 2026, Jelsoft Enterprises Ltd.
Served by www3v