Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Micron e millisecondi: la piattaforma ServiceNow guida l'infrastruttura IT di Aston Martin F1
Micron e millisecondi: la piattaforma ServiceNow guida l'infrastruttura IT di Aston Martin F1
Dalla precisione estrema dei componenti alla gestione digitale dei processi: come la piattaforma ServiceNow consente ad Aston Martin Aramco Formula One Team di ottimizzare risorse IT e flussi operativi in un ambiente dove ogni millesimo di secondo conta
ASUS GeForce RTX 5080 Noctua OC Edition: una custom fenomenale, ma anche enorme
ASUS GeForce RTX 5080 Noctua OC Edition: una custom fenomenale, ma anche enorme
ASUS e Noctua tornano a collaborare con la GeForce RTX 5080 Noctua OC Edition, una scheda pensata per chi cerca potenza estrema e silenziosità assoluta. Il nuovo sistema di raffreddamento, con tre ventole Noctua NF-A12x25 G2 da 120 mm e una camera di vapore maggiorata, promette temperature record e rumorosità quasi impercettibile. Non mancano dual BIOS, materiali di qualità e ampie possibilità di overclock. Ma quanto migliora davvero rispetto alla Founders Edition? Scoprilo nel nostro test completo.
Dreame Aqua10 Ultra Roller, la pulizia di casa con un rullo
Dreame Aqua10 Ultra Roller, la pulizia di casa con un rullo
Il più recente robot per la pulizia domestica di Dreame, modello Aqua10 Ultra Roller, abbina un potente motore di aspirazione della polvere a un sofisticato sistema di lavaggio con rullo integrato. Il tutto governato dalla logica di intelligenza artificiale, per i migliori risultati
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 15-03-2007, 13:05   #1
mfonz85
Member
 
L'Avatar di mfonz85
 
Iscritto dal: Sep 2005
Città: Bus PCI 1, periferica 0, funzione 0 (Torino)
Messaggi: 213
[C] Problema x programma di network su Unix

Ciao a tutti,
avrei un problema: devo far parlare un client e un server scritti in C che gireranno su Unix, e devo praticamente fare una sorta di handshaking iniziale dove il client manda un carattere di saluto (codificato in ASCII - 1 byte) seguito da un'altro numero, scritto sempre su 1 byte, che può assumere il valore 1, 2 o 4: questo valore serve praticamente a far capire al server che da quel momento io (client) inizierò a mandargli dei numeri interi codificati appunto su 1, 2 o 4 byte.

I problemi miei sono i seguenti:
1) ma per il little/big endian?? Non succedono casini quando invio dati binari?
2) per ricostruire lo stream al server, che si vede sostanzialmente arrivare uno stream di byte, come faccio??
3) E anche al client, come faccio a concatenare in modo intelligente questi byte??

Poniamo il caso che devo fare un programma così: handshaking con una "R" seguita dal numero 1, 2, o 4 scritto sempre su 1 byte. (byte di CODIFICA). Nel nostro esempio voglio che il client invii da quel momento in poi solo interi codificati su 2 byte.
Il server lo deve ricevere, e capire come aspettarsi i prossimi dati. (ovvero da adesso in poi si aspetta SOLO interi su 2 byte)
Poi da client voglio inviargli 4 numeri consecutivi codificati sul numero di byte scritto in precedenza sul secondo campo del primo invio (byte di CODIFICA)
Per esempio gli voglio inviare 67, 621 e 715 scritti su 2 byte. Questi numeri devono essere codificati su 2 byte, e messi in un'unico stream grosso di 4*2 = 8 byte (che saranno poi spezzettati al server)

Qualcuno mi può dare una mano a scrivere un pezzetto di codice che fa questo lavoretto??

Ringrazio anticipatamente!
__________________
Ho concluso affari con: Ippo 2001, Klintf, albert78, Piripikkio, starsky, oldfield e IL0V€INT€R. da EVITARE zarovat
mfonz85 è offline   Rispondi citando il messaggio o parte di esso
Old 15-03-2007, 18:02   #2
mfonz85
Member
 
L'Avatar di mfonz85
 
Iscritto dal: Sep 2005
Città: Bus PCI 1, periferica 0, funzione 0 (Torino)
Messaggi: 213
nessuna idea ?!?!
__________________
Ho concluso affari con: Ippo 2001, Klintf, albert78, Piripikkio, starsky, oldfield e IL0V€INT€R. da EVITARE zarovat
mfonz85 è offline   Rispondi citando il messaggio o parte di esso
Old 15-03-2007, 19:19   #3
cionci
Senior Member
 
L'Avatar di cionci
 
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
Quote:
Originariamente inviato da mfonz85 Guarda i messaggi
Ciao a tutti,
avrei un problema: devo far parlare un client e un server scritti in C che gireranno su Unix, e devo praticamente fare una sorta di handshaking iniziale dove il client manda un carattere di saluto (codificato in ASCII - 1 byte) seguito da un'altro numero, scritto sempre su 1 byte, che può assumere il valore 1, 2 o 4: questo valore serve praticamente a far capire al server che da quel momento io (client) inizierò a mandargli dei numeri interi codificati appunto su 1, 2 o 4 byte.

I problemi miei sono i seguenti:
1) ma per il little/big endian?? Non succedono casini quando invio dati binari?
Sì. Infatti, se non abbiamo problemi di "traffico", a volte quando si inviano molti dati strutturati si preferisce usare XML.
C'è comunque la possibilità di usare funzioni di libreria che rendono indipendenti i dati scritti dall'architettura: guarda il man di htonl e compagni.
Quote:
Originariamente inviato da mfonz85 Guarda i messaggi
2) per ricostruire lo stream al server, che si vede sostanzialmente arrivare uno stream di byte, come faccio??
Devi costruirti un protocollo. Il server deve sapere a priori la dimensione dai dati. Te lo spiego sotto.
Quote:
Originariamente inviato da mfonz85 Guarda i messaggi
3) E anche al client, come faccio a concatenare in modo intelligente questi byte??
Non importa, usa una struttura dati. Riempila con le debite conversioni hton* ed inviala per intero.
Quote:
Originariamente inviato da mfonz85 Guarda i messaggi
Poniamo il caso che devo fare un programma così: handshaking con una "R" seguita dal numero 1, 2, o 4 scritto sempre su 1 byte. (byte di CODIFICA). Nel nostro esempio voglio che il client invii da quel momento in poi solo interi codificati su 2 byte.
Il server lo deve ricevere, e capire come aspettarsi i prossimi dati. (ovvero da adesso in poi si aspetta SOLO interi su 2 byte)
Poi da client voglio inviargli 4 numeri consecutivi codificati sul numero di byte scritto in precedenza sul secondo campo del primo invio (byte di CODIFICA)
Per esempio gli voglio inviare 67, 621 e 715 scritti su 2 byte. Questi numeri devono essere codificati su 2 byte, e messi in un'unico stream grosso di 4*2 = 8 byte (che saranno poi spezzettati al server)
Hai descritto un protocollo. Il modo migliore per gestire un protocollo è quello di una macchina a stati, ad esempio (me lo invento):
avviene la connessione: passo allo stato IDENTIFYING
IDENTIFYING: devo ricevere username e password. La codifica dei dati deve essere composta da (in sequenza di lettura):
- 2 byte per la dimensione dello username
- stringa username
- 2 byte per la dimensione della password
- stringa password
se l'autenticazione ha successo passo a ENCODING
se l'autenticazione non ha successo passo a FAIL

ENCODING: devo ricevere R seguito da 1 byte, quindi leggo 2 byte.
Se le informazioni non sono corrette passo a FAIL
Se ricevo 1 passo a 1BYTE_ENCODED
Se ricevo 2 passo a 2BYTE_ENCODED
Se ricevo 4 passo a 4BYTE_ENCODED

1BYTE_ENCODED: leggo la dimensione dei dati da leggere, leggo un byte alla volta e faccio quello che ci devo fare
2BYTE_ENCODED: leggo la dimensione dei dati da leggere, leggo due byte alla volta e faccio quello che ci devo fare
4BYTE_ENCODED: leggo la dimensione dei dati da leggere, leggo quattro byte alla volta e faccio quello che ci devo fare

FAIL: rispondo con una stringa di errore (sempre dimensione, seguito dalla stringa). Vado a CLOSING

CLOSING: termino la connessione

La macchina a stati te la gestisci in maniera semplice con un enum, in base al valore dello stato sai quello che devi fare (il modo più immediato per implementarlo è uno switch),
cionci è offline   Rispondi citando il messaggio o parte di esso
Old 15-03-2007, 21:05   #4
mfonz85
Member
 
L'Avatar di mfonz85
 
Iscritto dal: Sep 2005
Città: Bus PCI 1, periferica 0, funzione 0 (Torino)
Messaggi: 213
Grazie, sei stato molto chiaro...ma vorrei farti ancora un paio di domande...

Quote:
Originariamente inviato da cionci Guarda i messaggi
Non importa, usa una struttura dati. Riempila con le debite conversioni hton* ed inviala per intero.
Quindi quando devo inviare al server l'handshaking, invio i dati inseriti in 1 roba del genere?

Codice:
struct _dati {
   char saluto;
   char codifica; //con valori possibili 1,2, o 4
} DATI;

DATI.saluto = 'R';
DATI.codifica = 2;

//problema: devo usare "uint16_t htons" (host to network short) per un carattere: ma si può fare?!?!?

(short int) DATI.saluto = htons((int) DATI.saluto); //ma si può fare?!?!?
(short int) DATI.codifica = htons((int) DATI.codifica); //ma si può fare?!?!?

//problema 2: con che funzione e come la invio la struttura dati?? Come un puntatore void * liscio??

... to do ... to do ... to do
Quote:
Originariamente inviato da cionci Guarda i messaggi
1BYTE_ENCODED: leggo la dimensione dei dati da leggere, leggo un byte alla volta e faccio quello che ci devo fare
2BYTE_ENCODED: leggo la dimensione dei dati da leggere, leggo due byte alla volta e faccio quello che ci devo fare
4BYTE_ENCODED: leggo la dimensione dei dati da leggere, leggo quattro byte alla volta e faccio quello che ci devo fare
E per leggere un byte alla volta?? Con htons mi arrivano dei dati interi su 2 byte...quindi 2 byte per volta...

Avrei bisogno di un pezzo di codice, un esempietto, per capire...
__________________
Ho concluso affari con: Ippo 2001, Klintf, albert78, Piripikkio, starsky, oldfield e IL0V€INT€R. da EVITARE zarovat
mfonz85 è offline   Rispondi citando il messaggio o parte di esso
Old 16-03-2007, 08:49   #5
cionci
Senior Member
 
L'Avatar di cionci
 
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
Per i char non devi usare funzioni di conversione....

send(s, &DATO, sizeof(DATO), 0);

In ricezione puoi anche semplicemente leggere un byte alla volta oppure una struttura dati intera:

recv(s, &DATO, sizeof(DATO), 0);
Attenzione nei messaggi lunghi, la recv può anche ritornare prima di aver letto sizeof(DATO).

Come vedi sono perfettamente speculari.
Ovviamente se DATO fosse una struttura più complessa dovresti usare le funzioni di conversione per i dati contenuti nella struttura di dimensione superiore a 1 byte
cionci è offline   Rispondi citando il messaggio o parte di esso
Old 16-03-2007, 13:09   #6
mfonz85
Member
 
L'Avatar di mfonz85
 
Iscritto dal: Sep 2005
Città: Bus PCI 1, periferica 0, funzione 0 (Torino)
Messaggi: 213
Quote:
Originariamente inviato da cionci Guarda i messaggi
Per i char non devi usare funzioni di conversione....

send(s, &DATO, sizeof(DATO), 0);
Quindi niente problemi little/big endian per caratteri singoli??
Quindi prendiamo per esempio il carattere H maiuscolo, codice ASCII 72, codifica binaria 10001000: lo invio con una send, senza hton o simili, e poi al server mi sparo una recv su 1 byte?

Quote:
Originariamente inviato da cionci Guarda i messaggi
In ricezione puoi anche semplicemente leggere un byte alla volta oppure una struttura dati intera:

recv(s, &DATO, sizeof(DATO), 0);
Attenzione nei messaggi lunghi, la recv può anche ritornare prima di aver letto sizeof(DATO).

Come vedi sono perfettamente speculari.
Ovviamente se DATO fosse una struttura più complessa dovresti usare le funzioni di conversione per i dati contenuti nella struttura di dimensione superiore a 1 byte
Ecco, è questo che non riesco a capire...se DATO fosse una struttura complessa...come faccio a crearmela al client e poi a smontarmela al server??
__________________
Ho concluso affari con: Ippo 2001, Klintf, albert78, Piripikkio, starsky, oldfield e IL0V€INT€R. da EVITARE zarovat
mfonz85 è offline   Rispondi citando il messaggio o parte di esso
Old 16-03-2007, 14:00   #7
cionci
Senior Member
 
L'Avatar di cionci
 
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
Fai send e recv come ho scritto sopra e riempi la struttura.

Sul sender fai hton* per ogni dato superiore al byte e sul receiver applichi la funzione inversa ntoh*.
cionci è offline   Rispondi citando il messaggio o parte di esso
Old 16-03-2007, 14:28   #8
mfonz85
Member
 
L'Avatar di mfonz85
 
Iscritto dal: Sep 2005
Città: Bus PCI 1, periferica 0, funzione 0 (Torino)
Messaggi: 213
Ok, vediamo se ho capito bene...

Codice:
//client-side Data Sending

struct _numeri_su_2_byte {
   uint_16 numeroA;
   uint_16 numeroB;
   uint_8 operatore;
} DUEBYTE;

int main(void)
{
   DUEBYTE *ptr;
   ptr = (DUEBYTE) malloc(sizeof(DUEBYTE));
   ptr.numeroA = htons(16); //primo operatore: 16
   ptr.numeroB = htons(57); //secondo operatore: 57
   ptr.operatore = htons(43) //codice ASCII per il '+'
   send(socket, &DUEBYTE, sizeof(DUEBYTE), 0);
}
E' tutto ok??

Al server quindi basta in questo caso fare una recv su 5 byte giusto??

Codice:
//server-side Data Receiving

struct _numeri_su_2_byte {
   uint_16 numeroA;
   uint_16 numeroB;
   uint_8 operatore;
} arrivato;

int main(void) 
{
   uint_16 A, B;
   char op;
   
   recv(socket, &arrivato, 5*sizeof(char), 0);
   A = (uint_16) ntohs(arrivato.numeroA);
   B = (uint_16) ntohs(arrivato.numeroB);
   op = (char) ntohs(arrivato.operatore);
}
Tutto ok anche qui??

Ci sono altre migliorie/ottimizzazioni/idee a riguardo, trattandosi si roba delicata di trasferimenti via ethernet??
Domanda: se al server non ricevessi proprio 5 byte, ma di meno (o di più) come mi comporto??
__________________
Ho concluso affari con: Ippo 2001, Klintf, albert78, Piripikkio, starsky, oldfield e IL0V€INT€R. da EVITARE zarovat
mfonz85 è offline   Rispondi citando il messaggio o parte di esso
Old 16-03-2007, 20:27   #9
mfonz85
Member
 
L'Avatar di mfonz85
 
Iscritto dal: Sep 2005
Città: Bus PCI 1, periferica 0, funzione 0 (Torino)
Messaggi: 213
Ma no, no, nooooo! Non funziona

sia che uso

Codice:
union DUEBYTE {
   u_int8_t operatore;
   u_int16_t numeroA;
   u_int16_t numeroB;
};
che

Codice:
typedef struct _numeri_su_2_byte {
   u_int8_t operatore;
   u_int16_t numeroA;
   u_int16_t numeroB;
} DUEBYTE;
quando vado a fare

Codice:
char op;
u_int16_t a,b;
DUEBYTE ptr;

if (sscanf(buf,"%d %c %d",&a, &op, &b) != 2) err_quit("(%s) error", prog);
else {
     ptr = malloc(sizeof(DUEBYTE));
     ptr.numeroA = htons(a); //primo operatore
     ptr.numeroB = htons(b); //secondo operatore
     ptr.operatore = htons(op); //codice ASCII per l'op
     send(sockfd, &DUEBYTE, sizeof(DUEBYTE), 0);
}
gcc, nel primo caso (union) mi fa:
DUEBYTE undeclared, ptr udneclared.

sempre gcc, nel secondo caso (typedef struct) mi fa:
request for member numeroA, numeroB, operatore in something not a structure or union...
Ma come?!?! Cosa vuole di più dalla vita il gcc? E' tutto perfetto!!
E oltretutto sto guardando una guida, mica mi sto inventando nulla: http://www.codeguru.com/forum/showthread.php?t=306399
__________________
Ho concluso affari con: Ippo 2001, Klintf, albert78, Piripikkio, starsky, oldfield e IL0V€INT€R. da EVITARE zarovat
mfonz85 è offline   Rispondi citando il messaggio o parte di esso
Old 16-03-2007, 20:42   #10
cionci
Senior Member
 
L'Avatar di cionci
 
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
Non usare la union...
Codice:
char op;
u_int16_t a,b;
DUEBYTE *ptr;

if (sscanf(buf,"%d %c %d",&a, &op, &b) != 2) err_quit("(%s) error", prog);
else {
     ptr = malloc(sizeof(DUEBYTE));
     ptr->numeroA = htons(a); //primo operatore
     ptr->numeroB = htons(b); //secondo operatore
     ptr->operatore = htons(op); //codice ASCII per l'op
     send(sockfd, ptr, sizeof(DUEBYTE), 0);
}
cionci è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


Micron e millisecondi: la piattaforma ServiceNow guida l'infrastruttura IT di Aston Martin F1 Micron e millisecondi: la piattaforma ServiceNow...
ASUS GeForce RTX 5080 Noctua OC Edition: una custom fenomenale, ma anche enorme ASUS GeForce RTX 5080 Noctua OC Edition: una cus...
Dreame Aqua10 Ultra Roller, la pulizia di casa con un rullo Dreame Aqua10 Ultra Roller, la pulizia di casa c...
Recensione Realme 15 Pro Game Of Thrones: un vero cimelio tech per pochi eletti Recensione Realme 15 Pro Game Of Thrones: un ver...
GIGABYTE GAMING A16, Raptor Lake e RTX 5060 Laptop insieme per giocare al giusto prezzo GIGABYTE GAMING A16, Raptor Lake e RTX 5060 Lapt...
La PS5 continua a vendere bene: la conso...
Colpo clamoroso della Cina: convalidato ...
Sabotati dall'interno: il flop di MindsE...
Intel Arc, il futuro è nebuloso: ...
TikTok cambia ancora le regole sulla pri...
The Outer Worlds 2: RPG vecchio stile, o...
1,9 miliardi in fumo (per ora): l'attacc...
Gli utenti di Steam possono monitorare l...
Dal digitale all'auto: HONOR e BYD insie...
Noctua compie 20 anni e festeggia con un...
Atari 2600+ PAC-MAN Edition torna in gra...
Xiaomi TV F Pro 75'' a 599€: il maxi sch...
Nissan Sakura si ricarica da sola: arriv...
PS6 non potrà puntare solo sulla ...
Project Bromo, Europa unita nello spazio...
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: 19:35.


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